Python 3でバイトを16進文字列に変換する正しい方法は何ですか?


236

Python 3でバイトを16進文字列に変換する正しい方法は何ですか?

私はbytes.hexメソッド、bytes.decodeコーデックの主張を目にし、他に驚くことのない他の可能な機能を試しましたが、役に立ちませんでした。私はバイトを16進数として欲しいだけです!


「無用」?どのような特定の問題またはエラーが発生していますか?コードとエラーを表示してください。
S.Lott、2011

回答:


410

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


5
bytes.fromhex()Python 3.0以降でも使用できます(3.5以降だけではありません)。bytes.hex()Python 3.5以降のみです。
フェニックス

95

binasciiモジュールを使用します。

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

この回答を参照してください: 16進数へのPython 3.1.1文字列


8
これはいい。気が遠くなると、bytes.fromhex(hex_str)を使用して16進数をバイトに変換できますが、bytes.tohex()を使用してバイトを16進数に変換することはできません。
nagylzs 2013年

1
バイトと16進数の関係はどちらの特性でもないようです(fromhexが存在する理由はわかりません)。それは単なる見落としではなく、議論されたもののようです:bugs.python.org/issue3532#msg70950。Q:このタスクを実行するために、bytesオブジェクトのtohexメソッドを使用することは害になりますか?A:IMO、そうです。コードが複雑になり、データ変換への適切なアプローチ(つまり、メソッドではなく関数)から焦点が外れます。
Mu Mind

3
これは本当に質問に答えますか?これは、六角戻りませんstrけどbytes。OPはその答えに満足しているように見えますが、この答えを拡張.decode("ascii")して「文字列」に変換することも含めた方がいいでしょう
RubenLaguna

3
多くの人がをプリントアウトする方法を探してこの質問/回答にたどり着くと思っていましたbytes。あなたならばprint(b'666f6f')、あなたは取得bプリントアウトに。もしそうなら.decode("ascii")、あなたはしません。実際にbytes(ASCII文字列ではなく、128より大きい項目を持つ真のバイナリ)を持っている人がそれをどのように印刷したかを考えただけです。
RubenLaguna

5
@nagylzs:.hex()Python 3.5以降にメソッドがある
jfs

43

Pythonには、quoted-printable(7ビットASCIIに適合する)、base64(英数字に適合する)、16進エスケープ、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、2014年

@JanusTroelsen:'hex_codec'を試してください。または、binascii.hexlify(b'foo')直接使用する
jfs 2014年


6

メソッドbinascii.hexlify()はASCIIの16進数文字列を表すに変換さbytesbytesます。つまり、入力の各バイトは2つの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

16進数文字列から"0a160a04"に戻って来ることができるとbytesしてbinascii.unhexlify("0a160a04")いるバックを与えますb'\n\x16\n\x04'


3

OK、次の答えは、Python 3のみに関心がある場合は少し範囲外ですが、Pythonのバージョンを指定しなくても、この質問が最初のGoogleヒットになるため、Python 2 Python 2の両方で機能する方法を次に示します Python 3の。

私はまた、バイトを変換することについての質問を解釈しています str型に:つまり、Python 2ではバイト-y、Python 3ではユニコード-yです。

それを考えると、私が知っている最善のアプローチは次のとおりです。

import six

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

次のアサーションはunicode_literals、Python 2で将来をアクティブ化していない場合、Python 2またはPython 3のいずれにも当てはまります。

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

(または''.join()、バイト間のスペースなどを省略するために使用できます)


3

%x0216進値をフォーマットして出力するフォーマット指定子を使用できます。例えば:

>>> 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()
BrunoL。


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
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.