Превратите шестнадцатеричную строку в строку, закодированную в процентах в Python

У меня есть строка. Это выглядит как s = 'e6b693e6a0abe699ab',

Я хочу поставить знак процента перед каждой парой символов, поэтому percentEncode(s) == '%e6%b6%93%e6%a0%ab%e6%99%ab',

Какой хороший способ написания percentEncode(s)?

(Обратите внимание, мне все равно, что незарезервированные символы не конвертируются в ASCII.)

Я могу придумать большие подробные способы сделать это, но я хочу что-то приятное и простое, и хотя я довольно новичок в Python, я был бы удивлен, если Python не может сделать это красиво.

6 ответов

Решение
>>> ''.join( "%"+i+s[n+1] for n,i in enumerate(s)  if n%2==0 )
'%e6%b6%93%e6%a0%ab%e6%99%ab'

Или используя ре

>>> import re
>>> re.sub("(..)","%\\1",s)
'%e6%b6%93%e6%a0%ab%e6%99%ab'

Если вы вручную выполняете кодировку URL-адресов, возможно, вы захотите прочитать этот пост в блоге. Это объясняет, как сделать это с помощью стандартной библиотеки urllib модуля quote_plus функция.

О, ты имеешь в виду:

''.join(["%%%s" % pair for pair in [s[i:i+2] for i in range(0,len(s),2)]])

Хотя, возможно, если вы делаете это для экранирования URL или чего-то подобного, есть библиотечная функция, более подходящая для вашего использования.

Отредактировано, чтобы добавить - так как все любят симпатичное решение itertools:

>>> from itertools import izip, cycle
>>> its = iter(s)
>>> tups = izip(cycle('%'), its, its)
>>> ''.join(''.join(t) for t in tups)
'%e6%b6%93%e6%a0%ab%e6%99%ab'

Просто чтобы быть академичным.

Попытка использовать как можно больше итераторов.

s = 'e6b693e6a0abe699ab'

from itertools import islice, izip, cycle, chain

def percentEncode(s):
    percentChars = cycle('%')
    firstChars = islice(s,0,None, 2)
    secondChars = islice(s,1,None, 2)
    return ''.join(chain.from_iterable(izip(percentChars, firstChars, secondChars)))


if __name__ == '__main__':
     print percentEncode(s)

Спасибо @tcarobruce за напоминание о необходимости повторного использования строки.

s = 'e6b693e6a0abe699ab'

from itertools import islice, izip, cycle, chain

def percentEncode(s):
    iter_s = iter(s)
    return ''.join(chain.from_iterable(izip(cycle('%'), iter_s, iter_s)))

if __name__ == '__main__':
     print percentEncode(s)

На основании вашего комментария в начальном вопросе, если вы начинаете с начальной строки initial_s перед его кодированием в hex вы можете получить результат как:

def percent_encode(initial_s):
    return ''.join('%%%02x' % ord(c) for c in initial_s)

>>> percent_encode('hello')
'%68%65%6c%6c%6f'

Использовать регулярное выражение для эффекта /([0-9a-f]{2})/ig и заменить на %\1

Другие вопросы по тегам