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

我看到一个字节的声明。十六进制方法,bytes.decode编解码器,并尝试了其他可能的最小惊讶函数,但没有任何效果。我只想要我的字节为十六进制!


当前回答

好吧,如果你只关心Python 3,下面的答案有点超出范围,但即使你没有指定Python版本,这个问题也是谷歌的第一个问题,所以这里有一种在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或Python 3都是正确的,假设你没有在Python 2中激活unicode_literals未来:

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

(或者你可以使用" .join()来省略字节之间的空格,等等)

其他回答

好吧,如果你只关心Python 3,下面的答案有点超出范围,但即使你没有指定Python版本,这个问题也是谷歌的第一个问题,所以这里有一种在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或Python 3都是正确的,假设你没有在Python 2中激活unicode_literals未来:

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

(或者你可以使用" .join()来省略字节之间的空格,等等)

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

使用binascii模块:

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

请看这个答案: Python 3.1.1字符串到十六进制

方法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")返回字节,返回b'\n\x16\n\x04'

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

适用于Python 3.3(因此“hex_codec”而不是“hex”)。