Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cpython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
cpython
Commits
9f0cbf1c
Commit
9f0cbf1c
authored
Sep 13, 2010
by
Daniel Stutzbach
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Issue #9213: Add index and count methods to range objects, needed to
meet the API of the collections.Sequence ABC.
parent
e4d6317c
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
206 additions
and
51 deletions
+206
-51
Doc/library/stdtypes.rst
Doc/library/stdtypes.rst
+16
-2
Lib/test/test_builtin.py
Lib/test/test_builtin.py
+54
-0
Misc/NEWS
Misc/NEWS
+4
-0
Objects/rangeobject.c
Objects/rangeobject.c
+132
-49
No files found.
Doc/library/stdtypes.rst
View file @
9f0cbf1c
...
...
@@ -1554,9 +1554,23 @@ looping. The advantage of the :class:`range` type is that an :class:`range`
object will always take the same amount of memory, no matter the size of the
range it represents. There are no consistent performance advantages.
Range objects have
very little behavior: they only support indexing, iteration
,
and the :func:`len` function
.
Range objects have
relatively little behavior: they support indexing
,
iteration, the :func:`len` function, and the following methods
.
.. method:: range.count(x)
Return the number of *i*'s for which ``s[i] == x``. Normally the
result will be 0 or 1, but it could be greater if *x* defines an
unusual equality function.
.. versionadded:: 3.2
.. method:: range.index(x)
Return the smallest *i* such that ``s[i] == x``. Raises
:exc:`ValueError` when *x* is not in the range.
.. versionadded:: 3.2
.. _typesseq-mutable:
...
...
Lib/test/test_builtin.py
View file @
9f0cbf1c
...
...
@@ -1028,6 +1028,60 @@ class BuiltinTest(unittest.TestCase):
self
.
assertRaises
(
TypeError
,
range
,
0.0
,
0.0
,
1
)
self
.
assertRaises
(
TypeError
,
range
,
0.0
,
0.0
,
1.0
)
self
.
assertEqual
(
range
(
3
).
count
(
-
1
),
0
)
self
.
assertEqual
(
range
(
3
).
count
(
0
),
1
)
self
.
assertEqual
(
range
(
3
).
count
(
1
),
1
)
self
.
assertEqual
(
range
(
3
).
count
(
2
),
1
)
self
.
assertEqual
(
range
(
3
).
count
(
3
),
0
)
self
.
assertEqual
(
range
(
10
**
20
).
count
(
1
),
1
)
self
.
assertEqual
(
range
(
10
**
20
).
count
(
10
**
20
),
0
)
self
.
assertEqual
(
range
(
3
).
index
(
1
),
1
)
self
.
assertEqual
(
range
(
1
,
2
**
100
,
2
).
count
(
2
**
87
),
0
)
self
.
assertEqual
(
range
(
1
,
2
**
100
,
2
).
count
(
2
**
87
+
1
),
1
)
self
.
assertEqual
(
range
(
1
,
10
,
3
).
index
(
4
),
1
)
self
.
assertEqual
(
range
(
1
,
-
10
,
-
3
).
index
(
-
5
),
2
)
self
.
assertEqual
(
range
(
10
**
20
).
index
(
1
),
1
)
self
.
assertEqual
(
range
(
10
**
20
).
index
(
10
**
20
-
1
),
10
**
20
-
1
)
self
.
assertRaises
(
ValueError
,
range
(
1
,
2
**
100
,
2
).
index
,
2
**
87
)
self
.
assertEqual
(
range
(
1
,
2
**
100
,
2
).
index
(
2
**
87
+
1
),
2
**
86
)
class
AlwaysEqual
(
object
):
def
__eq__
(
self
,
other
):
return
True
always_equal
=
AlwaysEqual
()
self
.
assertEqual
(
range
(
10
).
count
(
always_equal
),
10
)
self
.
assertEqual
(
range
(
10
).
index
(
always_equal
),
0
)
def
test_range_index
(
self
):
u
=
range
(
2
)
self
.
assertEqual
(
u
.
index
(
0
),
0
)
self
.
assertEqual
(
u
.
index
(
1
),
1
)
self
.
assertRaises
(
ValueError
,
u
.
index
,
2
)
u
=
range
(
-
2
,
3
)
self
.
assertEqual
(
u
.
count
(
0
),
1
)
self
.
assertEqual
(
u
.
index
(
0
),
2
)
self
.
assertRaises
(
TypeError
,
u
.
index
)
class
BadExc
(
Exception
):
pass
class
BadCmp
:
def
__eq__
(
self
,
other
):
if
other
==
2
:
raise
BadExc
()
return
False
a
=
range
(
4
)
self
.
assertRaises
(
BadExc
,
a
.
index
,
BadCmp
())
a
=
range
(
-
2
,
3
)
self
.
assertEqual
(
a
.
index
(
0
),
2
)
def
test_input
(
self
):
self
.
write_testfile
()
fp
=
open
(
TESTFN
,
'r'
)
...
...
Misc/NEWS
View file @
9f0cbf1c
...
...
@@ -10,6 +10,10 @@ What's New in Python 3.2 Alpha 3?
Core and Builtins
-----------------
- Issue #9212: The range type_items now provides index() and count()
methods, to conform to the Sequence ABC. Patch by Daniel Urban and
Daniel Stutzbach.
- Issue #7994: Issue a PendingDeprecationWarning if object.__format__
is called with a non-empty format string. This is an effort to
future-proof user code. If a derived class does not currently
...
...
Objects/rangeobject.c
View file @
9f0cbf1c
...
...
@@ -273,58 +273,133 @@ range_reduce(rangeobject *r, PyObject *args)
r
->
start
,
r
->
stop
,
r
->
step
);
}
/* Assumes (PyLong_CheckExact(ob) || PyBool_Check(ob)) */
static
int
range_contains_long
(
rangeobject
*
r
,
PyObject
*
ob
)
{
int
cmp1
,
cmp2
,
cmp3
;
PyObject
*
tmp1
=
NULL
;
PyObject
*
tmp2
=
NULL
;
PyObject
*
zero
=
NULL
;
int
result
=
-
1
;
zero
=
PyLong_FromLong
(
0
);
if
(
zero
==
NULL
)
/* MemoryError in int(0) */
goto
end
;
/* Check if the value can possibly be in the range. */
cmp1
=
PyObject_RichCompareBool
(
r
->
step
,
zero
,
Py_GT
);
if
(
cmp1
==
-
1
)
goto
end
;
if
(
cmp1
==
1
)
{
/* positive steps: start <= ob < stop */
cmp2
=
PyObject_RichCompareBool
(
r
->
start
,
ob
,
Py_LE
);
cmp3
=
PyObject_RichCompareBool
(
ob
,
r
->
stop
,
Py_LT
);
}
else
{
/* negative steps: stop < ob <= start */
cmp2
=
PyObject_RichCompareBool
(
ob
,
r
->
start
,
Py_LE
);
cmp3
=
PyObject_RichCompareBool
(
r
->
stop
,
ob
,
Py_LT
);
}
if
(
cmp2
==
-
1
||
cmp3
==
-
1
)
/* TypeError */
goto
end
;
if
(
cmp2
==
0
||
cmp3
==
0
)
{
/* ob outside of range */
result
=
0
;
goto
end
;
}
/* Check that the stride does not invalidate ob's membership. */
tmp1
=
PyNumber_Subtract
(
ob
,
r
->
start
);
if
(
tmp1
==
NULL
)
goto
end
;
tmp2
=
PyNumber_Remainder
(
tmp1
,
r
->
step
);
if
(
tmp2
==
NULL
)
goto
end
;
/* result = (int(ob) - start % step) == 0 */
result
=
PyObject_RichCompareBool
(
tmp2
,
zero
,
Py_EQ
);
end:
Py_XDECREF
(
tmp1
);
Py_XDECREF
(
tmp2
);
Py_XDECREF
(
zero
);
return
result
;
}
static
int
range_contains
(
rangeobject
*
r
,
PyObject
*
ob
)
{
if
(
PyLong_CheckExact
(
ob
)
||
PyBool_Check
(
ob
))
return
range_contains_long
(
r
,
ob
);
return
(
int
)
_PySequence_IterSearch
((
PyObject
*
)
r
,
ob
,
PY_ITERSEARCH_CONTAINS
);
}
static
PyObject
*
range_count
(
rangeobject
*
r
,
PyObject
*
ob
)
{
if
(
PyLong_CheckExact
(
ob
)
||
PyBool_Check
(
ob
))
{
int
cmp1
,
cmp2
,
cmp3
;
PyObject
*
tmp1
=
NULL
;
PyObject
*
tmp2
=
NULL
;
PyObject
*
zero
=
NULL
;
int
result
=
-
1
;
zero
=
PyLong_FromLong
(
0
);
if
(
zero
==
NULL
)
/* MemoryError in int(0) */
goto
end
;
/* Check if the value can possibly be in the range. */
cmp1
=
PyObject_RichCompareBool
(
r
->
step
,
zero
,
Py_GT
);
if
(
cmp1
==
-
1
)
goto
end
;
if
(
cmp1
==
1
)
{
/* positive steps: start <= ob < stop */
cmp2
=
PyObject_RichCompareBool
(
r
->
start
,
ob
,
Py_LE
);
cmp3
=
PyObject_RichCompareBool
(
ob
,
r
->
stop
,
Py_LT
);
}
else
{
/* negative steps: stop < ob <= start */
cmp2
=
PyObject_RichCompareBool
(
ob
,
r
->
start
,
Py_LE
);
cmp3
=
PyObject_RichCompareBool
(
r
->
stop
,
ob
,
Py_LT
);
}
if
(
range_contains_long
(
r
,
ob
))
Py_RETURN_TRUE
;
else
Py_RETURN_FALSE
;
}
else
{
Py_ssize_t
count
;
count
=
_PySequence_IterSearch
((
PyObject
*
)
r
,
ob
,
PY_ITERSEARCH_COUNT
);
if
(
count
==
-
1
)
return
NULL
;
return
PyLong_FromSsize_t
(
count
);
}
}
if
(
cmp2
==
-
1
||
cmp3
==
-
1
)
/* TypeError */
goto
end
;
if
(
cmp2
==
0
||
cmp3
==
0
)
{
/* ob outside of range */
result
=
0
;
goto
end
;
}
static
PyObject
*
range_index
(
rangeobject
*
r
,
PyObject
*
ob
)
{
PyObject
*
idx
,
*
tmp
;
int
contains
;
PyObject
*
format_tuple
,
*
err_string
;
static
PyObject
*
err_format
=
NULL
;
if
(
!
PyLong_CheckExact
(
ob
)
&&
!
PyBool_Check
(
ob
))
{
Py_ssize_t
index
;
index
=
_PySequence_IterSearch
((
PyObject
*
)
r
,
ob
,
PY_ITERSEARCH_INDEX
);
if
(
index
==
-
1
)
return
NULL
;
return
PyLong_FromSsize_t
(
index
);
}
contains
=
range_contains_long
(
r
,
ob
);
if
(
contains
==
-
1
)
return
NULL
;
if
(
!
contains
)
goto
value_error
;
tmp
=
PyNumber_Subtract
(
ob
,
r
->
start
);
if
(
tmp
==
NULL
)
return
NULL
;
/* idx = (ob - r.start) // r.step */
idx
=
PyNumber_FloorDivide
(
tmp
,
r
->
step
);
Py_DECREF
(
tmp
);
return
idx
;
value_error:
/* Check that the stride does not invalidate ob's membership. */
tmp1
=
PyNumber_Subtract
(
ob
,
r
->
start
);
if
(
tmp1
==
NULL
)
goto
end
;
tmp2
=
PyNumber_Remainder
(
tmp1
,
r
->
step
);
if
(
tmp2
==
NULL
)
goto
end
;
/* result = (int(ob) - start % step) == 0 */
result
=
PyObject_RichCompareBool
(
tmp2
,
zero
,
Py_EQ
);
end:
Py_XDECREF
(
tmp1
);
Py_XDECREF
(
tmp2
);
Py_XDECREF
(
zero
);
return
result
;
/* object is not in the range */
if
(
err_format
==
NULL
)
{
err_format
=
PyUnicode_FromString
(
"%r is not in range"
);
if
(
err_format
==
NULL
)
return
NULL
;
}
/* Fall back to iterative search. */
return
(
int
)
_PySequence_IterSearch
((
PyObject
*
)
r
,
ob
,
PY_ITERSEARCH_CONTAINS
);
format_tuple
=
PyTuple_Pack
(
1
,
ob
);
if
(
format_tuple
==
NULL
)
return
NULL
;
err_string
=
PyUnicode_Format
(
err_format
,
format_tuple
);
Py_DECREF
(
format_tuple
);
if
(
err_string
==
NULL
)
return
NULL
;
PyErr_SetObject
(
PyExc_ValueError
,
err_string
);
Py_DECREF
(
err_string
);
return
NULL
;
}
static
PySequenceMethods
range_as_sequence
=
{
...
...
@@ -344,10 +419,18 @@ static PyObject * range_reverse(PyObject *seq);
PyDoc_STRVAR
(
reverse_doc
,
"Returns a reverse iterator."
);
PyDoc_STRVAR
(
count_doc
,
"rangeobject.count(value) -> integer -- return number of occurrences of value"
);
PyDoc_STRVAR
(
index_doc
,
"rangeobject.index(value, [start, [stop]]) -> integer -- return index of value.
\n
"
"Raises ValueError if the value is not present."
);
static
PyMethodDef
range_methods
[]
=
{
{
"__reversed__"
,
(
PyCFunction
)
range_reverse
,
METH_NOARGS
,
reverse_doc
},
{
"__reduce__"
,
(
PyCFunction
)
range_reduce
,
METH_VARARGS
},
{
"__reversed__"
,
(
PyCFunction
)
range_reverse
,
METH_NOARGS
,
reverse_doc
},
{
"__reduce__"
,
(
PyCFunction
)
range_reduce
,
METH_VARARGS
},
{
"count"
,
(
PyCFunction
)
range_count
,
METH_O
,
count_doc
},
{
"index"
,
(
PyCFunction
)
range_index
,
METH_O
,
index_doc
},
{
NULL
,
NULL
}
/* sentinel */
};
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment