summaryrefslogtreecommitdiffstats
path: root/Objects
diff options
context:
space:
mode:
authorAndrew Dalke <dalke@dalkescientific.com>2006-05-26 20:25:22 (GMT)
committerAndrew Dalke <dalke@dalkescientific.com>2006-05-26 20:25:22 (GMT)
commit51324078680cf636218866ba822b89c149ce5e15 (patch)
tree7e5b925d900fbdf061c97d0a2f80bc92b021f43c /Objects
parente4e023c4d3104cbf144437d25e6906e828a28993 (diff)
downloadcpython-51324078680cf636218866ba822b89c149ce5e15.zip
cpython-51324078680cf636218866ba822b89c149ce5e15.tar.gz
cpython-51324078680cf636218866ba822b89c149ce5e15.tar.bz2
Added limits to the replace code so it does not count all of the matching
patterns in a string, only the number needed by the max limit.
Diffstat (limited to 'Objects')
-rw-r--r--Objects/stringobject.c41
1 files changed, 19 insertions, 22 deletions
diff --git a/Objects/stringobject.c b/Objects/stringobject.c
index 8d2a0be..c59c74e 100644
--- a/Objects/stringobject.c
+++ b/Objects/stringobject.c
@@ -2477,7 +2477,7 @@ return_self(PyStringObject *self)
}
Py_LOCAL(Py_ssize_t)
-countchar(char *target, int target_len, char c)
+ countchar(char *target, int target_len, char c, Py_ssize_t maxcount)
{
Py_ssize_t count=0;
char *start=target;
@@ -2485,9 +2485,10 @@ countchar(char *target, int target_len, char c)
while ( (start=findchar(start, end-start, c)) != NULL ) {
count++;
+ if (count >= maxcount)
+ break;
start += 1;
}
-
return count;
}
@@ -2534,7 +2535,7 @@ countstring(char *target, Py_ssize_t target_len,
char *pattern, Py_ssize_t pattern_len,
Py_ssize_t start,
Py_ssize_t end,
- int direction)
+ int direction, Py_ssize_t maxcount)
{
Py_ssize_t count=0;
@@ -2552,21 +2553,26 @@ countstring(char *target, Py_ssize_t target_len,
}
/* zero-length substrings match everywhere */
- if (pattern_len == 0)
- return target_len+1;
+ if (pattern_len == 0 || maxcount == 0) {
+ if (target_len+1 < maxcount)
+ return target_len+1;
+ return maxcount;
+ }
end -= pattern_len;
-
if (direction < 0) {
- for (; end >= start; end--)
+ for (; (end >= start); end--)
if (Py_STRING_MATCH(target, end, pattern, pattern_len)) {
count++;
+ if (--maxcount <= 0) break;
end -= pattern_len-1;
}
} else {
- for (; start <= end; start++)
+ for (; (start <= end); start++)
if (Py_STRING_MATCH(target, start, pattern, pattern_len)) {
count++;
+ if (--maxcount <= 0)
+ break;
start += pattern_len-1;
}
}
@@ -2653,12 +2659,10 @@ replace_delete_single_character(PyStringObject *self,
self_len = PyString_GET_SIZE(self);
self_s = PyString_AS_STRING(self);
- count = countchar(self_s, self_len, from_c);
+ count = countchar(self_s, self_len, from_c, maxcount);
if (count == 0) {
return return_self(self);
}
- if (count > maxcount)
- count = maxcount;
result_len = self_len - count; /* from_len == 1 */
assert(result_len>=0);
@@ -2701,10 +2705,8 @@ replace_delete_substring(PyStringObject *self, PyStringObject *from,
count = countstring(self_s, self_len,
from_s, from_len,
- 0, self_len, 1);
-
- if (count > maxcount)
- count = maxcount;
+ 0, self_len, 1,
+ maxcount);
if (count == 0) {
/* no matches */
@@ -2857,9 +2859,7 @@ replace_single_character(PyStringObject *self,
self_s = PyString_AS_STRING(self);
self_len = PyString_GET_SIZE(self);
- count = countchar(self_s, self_len, from_c);
- if (count > maxcount)
- count = maxcount;
+ count = countchar(self_s, self_len, from_c, maxcount);
if (count == 0) {
/* no matches, return unchanged */
@@ -2933,10 +2933,7 @@ replace_substring(PyStringObject *self,
count = countstring(self_s, self_len,
from_s, from_len,
- 0, self_len, FORWARD);
- if (count > maxcount)
- count = maxcount;
-
+ 0, self_len, FORWARD, maxcount);
if (count == 0) {
/* no matches, return unchanged */
return return_self(self);