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

Posted

技术标签:

【中文标题】在 Python 3 中将字节转换为十六进制字符串的正确方法是啥?【英文标题】:What's the correct way to convert bytes to a hex string in Python 3?在 Python 3 中将字节转换为十六进制字符串的正确方法是什么? 【发布时间】:2011-10-01 06:05:18 【问题描述】:

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

我看到了bytes.hex 方法、bytes.decode 编解码器的声明,并尝试了other 可能的最不令人惊讶的功能,但无济于事。我只希望我的字节为十六进制!

【问题讨论】:

“无济于事”?您遇到了哪些具体问题或错误?请显示代码和错误。 ***.com/questions/2340319/python-3-1-1-string-to-hex的可能重复 【参考方案1】:

从 Python 3.5 开始,这终于不再尴尬了:

>>> b'\xde\xad\xbe\xef'.hex()
'deadbeef'

反过来:

>>> bytes.fromhex('deadbeef')
b'\xde\xad\xbe\xef'

也适用于可变的bytearray 类型。

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

【讨论】:

bytes.fromhex() 也可用于 Python 3.0+(不仅仅是 3.5+)。 bytes.hex() 仅适用于 Python 3.5+。【参考方案2】:

使用binascii 模块:

>>> import binascii
>>> binascii.hexlify('foo'.encode('utf8'))
b'666f6f'
>>> binascii.unhexlify(_).decode('utf8')
'foo'

看到这个答案: Python 3.1.1 string to hex

【讨论】:

这很好。令人难以置信的是,您可以使用 bytes.fromhex(hex_str) 将十六进制转换为字节,但不能使用 bytes.tohex() 将字节转换为十六进制 - 这其中的原因是什么? 我猜字节和十六进制之间的关系不是两者的属性(这不能回答为什么 fromhex 存在)。似乎这不仅仅是一个疏忽,而是有人争论的事情:bugs.python.org/issue3532#msg70950。问:让 bytes 对象的 tohex 方法也执行此任务会不会很痛苦?答:IMO,是的,会的。它使代码复杂化,并将焦点从正确的数据转换方法(即函数 - 而不是方法)上移开。 这真的回答了这个问题吗?它不返回十六进制 str,而是返回 bytes。我知道 OP 似乎对答案很满意,但最好将这个答案扩展到包括 .decode("ascii") 并将其转换为“字符串” 我在想,很多人都会关注这个问题/答案,寻找一种打印出bytes 的方法。如果您print(b'666f6f'),您会在打印输出中获得b。如果你.decode("ascii"),那么你不会。只是想那些实际上有一个 bytes (真正的二进制与项目 > 128 ,而不是一个 ascii 字符串)想打印出来。 @nagylzs: Python 3.5+ 中有.hex() 方法【参考方案3】:

Python 具有字节到字节的standard codecs,可以执行方便的转换,例如带引号的打印(适合 7 位 ascii)、base64(适合字母数字)、十六进制转义、gzip 和 bz2 压缩。在 Python 2 中,您可以这样做:

b'foo'.encode('hex')

在 Python 3 中,str.encode / bytes.decode 严格用于字节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 不太一致,但对于压缩编解码器,它提供了更多控制。

【讨论】:

使用python 3.3:LookupError: unknown encoding: hex @JanusTroelsen:试试'hex_codec'。或者直接使用binascii.hexlify(b'foo')【参考方案4】:

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

【讨论】:

【参考方案5】:

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

我包含了一个说明它的 sn-p。

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" 可以返回到bytesbinascii.unhexlify("0a160a04"),它返回b'\n\x16\n\x04'

【讨论】:

【参考方案6】:
import codecs
codecs.getencoder('hex_codec')(b'foo')[0]

在 Python 3.3 中工作(所以“hex_codec”而不是“hex”)。

【讨论】:

也许有趣的是,在 Python 3.4 中,“hex”或“hex_codec”可以正常工作。【参考方案7】:

好的,如果您只关心 Python 3,那么下面的答案稍微超出了范围,但是即使您没有指定 Python 版本,这个问题也是 Google 的第一个热门问题,所以这是一种适用于 Python 2 的方法 Python 3。

我还将这个问题解释为关于将字节转换为 str 类型:即 Python 2 上的 bytes-y 和 Python 3 上的 Unicode-y。

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

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()省略字节之间的空格等)

【讨论】:

【参考方案8】:

可以使用格式说明符%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()【参考方案9】:

如果你想将 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'

参考:https://docs.python.org/3.5/library/struct.html

【讨论】:

不知何故帮助我处理 esp32 绝对是,正如@TejasTank 所说。这是唯一适用于大量库不可用的 MicroPhyton 的方法。

以上是关于在 Python 3 中将字节转换为十六进制字符串的正确方法是啥?的主要内容,如果未能解决你的问题,请参考以下文章

在Python 3中将十六进制字符串(python 2)转换为带有十六进制数字的字节

在 Python 中将十六进制字符串转换为字节列表

在python中将字节转换为位

在 Python 中将整数转换为 2 字节的十六进制值

如何在 Python 中将字符串转换为字节数组

在Python中将IP地址字符串转换为二进制