在Python 3中,将字节转换为十六进制字符串的正确方法是什么?

383

在Python 3中将字节转换为十六进制字符串的正确方法是什么?

我看到了关于bytes.hex方法和bytes.decode编解码器的说法,并尝试了other一些最不惊讶的函数,但都没有成功。我只想把我的字节转换成十六进制!


2
“without avail”? 您遇到了什么具体的问题或错误?请展示代码和错误信息。 - S.Lott
可能是 https://dev59.com/-nE95IYBdhLWcg3wXsuR 的重复问题。 - Mu Mind
9个回答

733

18
bytes.fromhex() 在 Python 3.0+ 版本中也可用(不仅限于 3.5+ 版本)。bytes.hex() 仅在 Python 3.5+ 版本中可用。 - phoenix

110
使用binascii模块:
>>> import binascii
>>> binascii.hexlify('foo'.encode('utf8'))
b'666f6f'
>>> binascii.unhexlify(_).decode('utf8')
'foo'

参考这个答案: Python 3.1.1 string to hex


11
这很好。令人费解的是,您可以使用“bytes.fromhex(hex_str)”将十六进制转换为字节,但是您不能使用“bytes.tohex()”将字节转换为十六进制 - 这其中的原理是什么? - nagylzs
1
我猜字节和十六进制之间的关系不是它们本身的属性(这并没有回答为什么有fromhex)。看起来这不仅仅是一个疏忽,而是一些人争论过的事情:http://bugs.python.org/issue3532#msg70950。 问:让bytes对象的tohex方法也执行这个任务会有什么影响吗? 答:在我看来,会有影响。它会使代码变得复杂,并将注意力从正确的数据转换方法(即函数 - 而不是方法)转移开来。 - Mu Mind
6
这是否真正回答了问题?它没有返回十六进制的str,而是bytes。我知道提问者似乎对答案感到满意,但将这个答案扩展以包括.decode("ascii")也将其转换为“string”,会不会更好呢? - RubenLaguna
4
我想许多人会查看这个问题/答案是为了寻找打印 bytes 的方法。如果你打印 b'666f6f',则会在打印输出中得到 b。如果使用 .decode("ascii"),则不会出现 b。我在思考如何打印实际具有 bytes(包含大于 128 的真正二进制值,而不是 ASCII 字符串)的人。 - RubenLaguna
9
Python 3.5+ 中有一个 .hex() 方法。 - jfs
显示剩余2条评论

52

Python有字节到字节的标准编解码器,可以执行方便的转换,如quoted-printable(适合7位ASCII),base64(适合字母数字),十六进制转义、gzip和bz2压缩。在Python 2中,您可以这样做:

b'foo'.encode('hex')

在Python 3中,str.encode / bytes.decode 严格用于 bytes<->str 转换。而你可以使用以下方法,它适用于 Python 2 和 Python 3(对反向操作,使用 s/encode/decode/g):

import codecs
codecs.getencoder('hex')(b'foo')[0]

从Python 3.4开始,有一个不那么笨拙的选项:

codecs.encode(b'foo', 'hex')

这些杂项编解码器也可以在它们自己的模块(base64、zlib、bz2、uu、quopri、binascii)中访问;API 不太一致,但对于压缩编解码器来说,它提供了更多的控制。


1
使用Python 3.3:LookupError: unknown encoding: hex - Janus Troelsen
1
@JanusTroelsen:尝试使用'hex_codec'。或者直接使用binascii.hexlify(b'foo') - jfs

32

Python 3.8新增了一个功能,您可以向hex函数传递分隔符参数,例如:

>>> value = b'\xf0\xf1\xf2'
>>> value.hex('-')
'f0-f1-f2'
>>> value.hex('_', 2)
'f0_f1f2'
>>> b'UUDDLRLRAB'.hex(' ', -4)
'55554444 4c524c52 4142'

https://docs.python.org/3/library/stdtypes.html#bytes.hex


1
这段代码很有用:[f'0x{val}' for val in value.hex(' ', 2).split()] 可以得到类似于 ['0x2600', '0x9c00', '0x1900', '0x0c75'] 的结果。 - Dennis Williamson

12

方法binascii.hexlify()将把bytes转换为表示ASCII十六进制字符串的bytes。这意味着输入中的每个字节都会转换为两个ASCII字符。如果您想要一个真正的str,则可以.decode("ascii")结果。

我包括了一个演示它的代码片段。

import binascii

with open("addressbook.bin", "rb") as f: # or any binary file like '/bin/ls'
    in_bytes = f.read()
    print(in_bytes) # b'\n\x16\n\x04'
    hex_bytes = binascii.hexlify(in_bytes) 
    print(hex_bytes) # b'0a160a04' which is twice as long as in_bytes
    hex_str = hex_bytes.decode("ascii")
    print(hex_str) # 0a160a04

从十六进制字符串"0a160a04"可以通过binascii.unhexlify("0a160a04")转换回bytes,返回值为b'\n\x16\n\x04'


7
import codecs
codecs.getencoder('hex_codec')(b'foo')[0]

在Python 3.3中有效(因此使用“hex_codec”而不是“hex”)。

或许有趣的是,在Python 3.4中,“hex”或“hex_codec”可以正常工作。 - Stephen Paulger

6

可以使用格式说明符%x02来格式化并输出一个十六进制值。例如:

>>> foo = b"tC\xfc}\x05i\x8d\x86\x05\xa5\xb4\xd3]Vd\x9cZ\x92~'6"
>>> res = ""
>>> for b in foo:
...     res += "%02x" % b
... 
>>> print(res)
7443fc7d05698d8605a5b4d35d56649c5a927e2736

根据我的经验,这是最好的答案,因为它适用于每个Python版本,并且不需要任何导入。然而,我更喜欢将十六进制字符串显示为大写 res.upper() - Bruno L.
不需要使用 upper()!您可以通过 xX 控制大小写:'%02X' % 79 == '4F' - vav

4

好的,如果您只关心Python 3,则以下答案略有超出范围,但是即使您不指定Python版本,此问题也是谷歌搜索的第一个结果,因此这里介绍一种适用于Python 2和Python 3的方法。

我还将解释该问题是关于将字节转换为str类型:在Python 2上是字节,而在Python 3上是Unicode。

鉴于此,我所知道的最好方法是:

import six

bytes_to_hex_str = lambda b: ' '.join('%02x' % i for i in six.iterbytes(b))

假设您没有在Python 2中激活unicode_literals未来,以下断言对于Python 2或Python 3都是正确的:
assert bytes_to_hex_str(b'jkl') == '6a 6b 6c'

(或者您可以使用''.join()来省略字节之间的空格等。)

我发现在支持Python 3.7时这非常有用,因为在bytes.hex中没有sep参数。在撰写本文时,3.7仍然是受支持的版本。 - theY4Kman

1
如果你想将 b'\x61' 转换成 97 或者 '0x61',可以尝试以下方法:
[python3.5]
>>>from struct import *
>>>temp=unpack('B',b'\x61')[0] ## convert bytes to unsigned int
97
>>>hex(temp) ##convert int to string which is hexadecimal expression
'0x61'

Reference:https://docs.python.org/3.5/library/struct.html


以某种方式帮助我处理esp32 - Tejas Tank
绝对的,就像@TejasTank所说的那样。这是在MicroPython上工作的唯一方法,因为很多库都不可用。 - leosok

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接