Я хочу эффективный способ добавить одну строку к другой в Python, кроме следующего.
var1 = "foo"
var2 = "bar"
var3 = var1 + var2
Есть ли хороший встроенный метод для использования?
Я хочу эффективный способ добавить одну строку к другой в Python, кроме следующего.
var1 = "foo"
var2 = "bar"
var3 = var1 + var2
Есть ли хороший встроенный метод для использования?
Ответы:
Если у вас есть только одна ссылка на строку, и вы конкатенируете другую строку до конца, CPython теперь использует это в особых случаях и пытается расширить строку на месте.
Конечным результатом является то, что операция амортизируется O (n).
например
s = ""
for i in range(n):
s+=str(i)
Раньше было O (n ^ 2), но теперь это O (n).
Из источника (bytesobject.c):
void
PyBytes_ConcatAndDel(register PyObject **pv, register PyObject *w)
{
PyBytes_Concat(pv, w);
Py_XDECREF(w);
}
/* The following function breaks the notion that strings are immutable:
it changes the size of a string. We get away with this only if there
is only one module referencing the object. You can also think of it
as creating a new string object and destroying the old one, only
more efficiently. In any case, don't use this if the string may
already be known to some other part of the code...
Note that if there's not enough memory to resize the string, the original
string object at *pv is deallocated, *pv is set to NULL, an "out of
memory" exception is set, and -1 is returned. Else (on success) 0 is
returned, and the value in *pv may or may not be the same as on input.
As always, an extra byte is allocated for a trailing \0 byte (newsize
does *not* include that), and a trailing \0 byte is stored.
*/
int
_PyBytes_Resize(PyObject **pv, Py_ssize_t newsize)
{
register PyObject *v;
register PyBytesObject *sv;
v = *pv;
if (!PyBytes_Check(v) || Py_REFCNT(v) != 1 || newsize < 0) {
*pv = 0;
Py_DECREF(v);
PyErr_BadInternalCall();
return -1;
}
/* XXX UNREF/NEWREF interface should be more symmetrical */
_Py_DEC_REFTOTAL;
_Py_ForgetReference(v);
*pv = (PyObject *)
PyObject_REALLOC((char *)v, PyBytesObject_SIZE + newsize);
if (*pv == NULL) {
PyObject_Del(v);
PyErr_NoMemory();
return -1;
}
_Py_NewReference(*pv);
sv = (PyBytesObject *) *pv;
Py_SIZE(sv) = newsize;
sv->ob_sval[newsize] = '\0';
sv->ob_shash = -1; /* invalidate cached hash value */
return 0;
}
Это достаточно просто проверить эмпирически.
$ python -m timeit -s "s = ''" "для i в xrange (10): s + = 'a'" 1000000 петель, лучшее из 3: 1,85 мксек на петлю $ python -m timeit -s "s = ''" "для i в xrange (100): s + = 'a'" 10000 петель, лучшее из 3: 16,8 усек за петлю $ python -m timeit -s "s = ''" "для i в xrange (1000): s + = 'a'" 10000 циклов, лучшее из 3: 158 циклов на цикл $ python -m timeit -s "s = ''" "для i в xrange (10000): s + = 'a'" 1000 циклов, лучшее из 3: 1,71 мсек на цикл $ python -m timeit -s "s = ''" "для i в xrange (100000): s + = 'a'" 10 циклов, лучшее из 3: 14,6 мсек на цикл $ python -m timeit -s "s = ''" "для i в xrange (1000000): s + = 'a'" 10 циклов, лучшее из 3: 173 мсек на цикл
Однако важно отметить, что эта оптимизация не является частью спецификации Python. Это только в реализации cPython, насколько я знаю. Например, то же эмпирическое тестирование на pypy или jython может показать более высокую производительность O (n ** 2).
$ pypy -m timeit -s "s = ''" "для i в xrange (10): s + = 'a'" 10000 петель, лучшее из 3: 90,8 мксек на петлю $ pypy -m timeit -s "s = ''" "для i в xrange (100): s + = 'a'" 1000 циклов, лучшее из 3: 896 циклов на цикл $ pypy -m timeit -s "s = ''" "для i в xrange (1000): s + = 'a'" 100 циклов, лучшее из 3: 9,03 мсек на цикл $ pypy -m timeit -s "s = ''" "для i в xrange (10000): s + = 'a'" 10 циклов, лучшее из 3: 89,5 мсек на цикл
Пока все хорошо, но потом,
$ pypy -m timeit -s "s = ''" "для i в xrange (100000): s + = 'a'" 10 циклов, лучшее из 3: 12,8 с на цикл
ой даже хуже, чем квадратичный. Так что pypy делает что-то, что хорошо работает с короткими строками, но плохо работает с большими строками.
PyString_ConcatAndDel
функцию, но включили комментарий для _PyString_Resize
. Кроме того, комментарий на самом деле не подтверждает ваши претензии в отношении Big-O
"".join(str_a, str_b)
Не преждевременно оптимизировать. Если у вас нет причин полагать, что существует узкое место в скорости, вызванное конкатенацией строк, тогда просто придерживайтесь +
и +=
:
s = 'foo'
s += 'bar'
s += 'baz'
Тем не менее, если вы стремитесь к чему-то похожему на Java StringBuilder, каноническая идиома Python - добавить элементы в список, а затем использовать str.join
для объединения их всех в конце:
l = []
l.append('foo')
l.append('bar')
l.append('baz')
s = ''.join(l)
str1 = "Hello"
str2 = "World"
newstr = " ".join((str1, str2))
Это объединяет str1 и str2 с пробелом в качестве разделителей. Вы также можете сделать "".join(str1, str2, ...)
. str.join()
занимает многократное повторение, поэтому вам нужно будет поместить строки в список или кортеж.
Это примерно так же эффективно, как и для встроенного метода.
Не.
То есть в большинстве случаев вам лучше генерировать всю строку за один раз, а не добавлять к существующей строке.
Например, не делайте: obj1.name + ":" + str(obj1.count)
Вместо этого: используйте "%s:%d" % (obj1.name, obj1.count)
Это будет легче читать и эффективнее.
"<div class='" + className + "' id='" + generateUniqueId() + "'>" + message_text + "</div>"
я нахожу менее читабельным и подверженным ошибкам"<div class='{classname}' id='{id}'>{message_text}</div>".format(classname=class_name, message_text=message_text, id=generateUniqueId())
Если вам нужно выполнить много операций добавления для создания большой строки, вы можете использовать StringIO или cStringIO. Интерфейс похож на файл. То есть: вам, write
чтобы добавить текст к нему.
Если вы просто добавляете две строки, просто используйте +
.
В принципе, без разницы. Единственная устойчивая тенденция заключается в том, что Python становится медленнее с каждой версией ... :(
%%timeit
x = []
for i in range(100000000): # xrange on Python 2.7
x.append('a')
x = ''.join(x)
Python 2.7
1 цикл, лучшее из 3: 7,34 с на цикл
Python 3.4
1 цикл, лучшее из 3: 7,99 с на цикл
Python 3.5
1 цикл, лучшее из 3: 8,48 с на цикл
Python 3.6
1 цикл, лучшее из 3: 9,93 с на цикл
%%timeit
x = ''
for i in range(100000000): # xrange on Python 2.7
x += 'a'
Python 2.7 :
1 цикл, лучшее из 3: 7,41 с на цикл
Python 3.4
1 цикл, лучшее из 3: 9,08 с на цикл
Python 3.5
1 цикл, лучшее из 3: 8,82 с на цикл
Python 3.6
1 цикл, лучшее из 3: 9,24 с на цикл
1.19 s
и 992 ms
соответственно на Python2.7
добавлять строки с функцией __add__
str = "Hello"
str2 = " World"
st = str.__add__(str2)
print(st)
Вывод
Hello World
str + str2
все еще короче.
a='foo'
b='baaz'
a.__add__(b)
out: 'foobaaz'
a.__add__(b)
идентично письму a+b
. Когда вы объединяете строки, используя +
оператор, Python вызывает __add__
метод для строки слева, передавая строку правой части в качестве параметра.
"foo" + "bar" + str(3)