Как заменить набор символов внутри другой строки в Python?
Вот что я пытаюсь сделать: допустим, у меня есть строка 'abcdefghijkl' и я хочу заменить 2-й символ конца (k) на A. Я получаю ошибка:
>>> aa = 'abcdefghijkl'
>>> print aa[-2]
k
>>> aa[-2]='A'
Traceback (most recent call last):
File "<pyshell#2>", line 1, in <module>
aa[-2]='A'
TypeError: 'str' object does not support item assignment
Итак, вопрос: есть ли элегантный способ заменить (заменить) символами строки внутри другой строки, начиная с указанной позиции? Что-то вроде:
# subst(whole_string,symbols_to_substiture_with,starting_position)
>>> print aa
abcdefghijkl
>>> aa = subst(aa,'A',-2)
>>> print aa
abcdefghijAl
Каким будет код не грубой силы для саба?
Если вы заменяете всегда одну и ту же позицию, вы можете сделать что-то вроде:
>>> s = s[0:-2] + "A" + s[-1:]
>>> print s
abcdefghijAl
В общем случае вы можете сделать:
>>> rindex = -2 #Second to the last letter
>>> s = s[0:rindex] + "A" + s[rindex+1:]
>>> print s
abcdefghijAl
Edit: очень общий случай, если вы просто хотите повторить одну букву в замене:
>>> s = "abcdefghijkl"
>>> repl_str = "A"
>>> rindex = -4 #Start at 4th character from the end
>>> repl = 3 #Replace 3 characters
>>> s = s[0:rindex] + (repl_str * repl) + s[rindex+repl:]
>>> print s
abcdefghAAAl
TypeError: 'str' object does not support item assignment
Этого и следовало ожидать - строки Python неизменяемы.
Один из способов - нарезать кусочками и кубиками. Примерно так:
>>> aa = 'abcdefghijkl'
>>> changed = aa[0:-2] + 'A' + aa[-1]
>>> print changed
abcdefghijAl
Результатом конкатенации, изменено
будет другая неизменяемая строка. Имейте в виду, что это не универсальное решение, которое подходит для всех сценариев замены.
Другой способ сделать это:
s = "abcdefghijkl"
l = list( s )
l[-2] = 'A'
s = "".join( l )
Строки в Python - это неизменяемые последовательности - очень похожи на кортежи. Вы можете создать изменяемый список из строки или кортежа, изменить соответствующие индексы и преобразовать список обратно в строку с помощью join
или в кортеж
с помощью конструктора кортежей.
РЕДАКТИРОВАТЬ: в случае строк вы можете использовать тип bytearray ( docs ) вместо list
, что делает возможным преобразование в строку с помощью str
конструктор.
Также вы можете реализовать свой собственный или использовать этот класс из стандартной библиотеки: http://docs.python.org/library/userdict.html#UserString.MutableString вот так:
>>> from UserString import MutableString
>>> s = MutableString( "zdfgzbdr " )
>>> s
'zdfgzbdr '
>>> s[1:5]
'dfgz'
>>> s[1:5] = "xxxx"
>>> s
'zxxxxbdr '
К сожалению - MutableString
устарел и недоступен в Py3k (я думаю, вы все равно можете написать свой собственный класс).
Чтобы ответить на комментарий другого сообщения:
Чтобы сделать трехсимвольную замену непосредственно с помощью строк (см. Также сообщение bytearray), вы можете выполнить разделение с помощью последовательности, особенно если вы не знаете позицию без поиска:
aa = 'abcdefghijkl'
replace = 'def'
withstring = 'QRS'
newstr,found,endpart = aa.partition(replace)
if found:
newstr+=withstring+endpart
print newstr
else:
print "%r string is not in %r" % (replace,aa)
В этом случае длина замены может не совпадать с исходной.
Вы должны разрезать его и создать новую строку, поскольку строки неизменяемы:
aa = aa[:-2] + 'A' + aa[-1:]