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


235

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

我看到了bytes.hex方法,bytes.decode编解码器的声明,并尝试了其他最小惊讶的可能功能,但没有用。我只希望字节为十六进制!


“无济于事”?您遇到什么具体问题或错误?请显示代码和错误。
S.Lott

Answers:



95

使用binascii模块:

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

查看此答案: Python 3.1.1字符串转换为十六进制


8
很好 令人费解的是,您可以使用bytes.fromhex(hex_str)将十六进制转换为字节,但是不能使用bytes.tohex()将字节转换为十六进制-这有什么道理?
nagylzs

1
我猜想字节和十六进制之间的关系都不是两者的属性(这不能回答为什么fromhex存在)。似乎这不仅是疏忽,而且还引起了争论:bugs.python.org/issue3532#msg70950。问:使用bytes对象的tohex方法也可以执行此任务吗?答:IMO,是的。它使代码复杂化,并将焦点从正确的数据转换方法(即函数而不是方法)上转移了出来。
Mu Mind

3
这真的回答了这个问题吗?它不返回十六进制,str而是返回bytes。我知道OP似乎对答案很满意,但将答案扩展为包括.decode("ascii")将其转换为“字符串”也不会更好
RubenLaguna 2015年

3
我当时以为很多人都在寻找这个问题/答案,以寻找一种打印出的方法bytes。如果您print(b'666f6f')获得b了打印输出。如果您.decode("ascii")不这样做。只要考虑一下那些实际上具有bytes(项目> 128的真正二进制文件,而不是ASCII字符串)的人想要打印出来的方法。
RubenLaguna 2015年

5
@nagylzs:.hex()Python 3.5+中有方法
jfs

43

Python具有逐个字节的标准编解码器,可以执行方便的转换,例如带引号的可打印(适合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的一致性较差,但对于压缩编解码器,它提供了更多控制权。


1
使用python 3.3:LookupError: unknown encoding: hex
Janus Troelsen

@JanusTroelsen:尝试使用“ hex_codec”。或者只是使用binascii.hexlify(b'foo')直接
JFS

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

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


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

6

该方法binascii.hexlify()将转换bytesbytes代表ascii十六进制字符串的字符串。这意味着输入中的每个字节将转换为两个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"到可以回来了bytesbinascii.unhexlify("0a160a04")该还给b'\n\x16\n\x04'


3

好的,如果您只关心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))

假设您尚未激活unicode_literalsPython 2 的未来,以下断言对于Python 2或Python 3都是正确的:

assert bytes_to_hex_str(b'jkl') == '6a 6b 6c'

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


3

可以使用格式说明符%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.


0

如果要将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以某种方式帮助我
Tejas Tank
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.