วิธีที่ถูกต้องในการแปลงไบต์เป็นสตริงเลขฐานสิบหกใน Python 3 คืออะไร


236

วิธีที่ถูกต้องในการแปลงไบต์เป็นสตริงเลขฐานสิบหกใน Python 3 คืออะไร

ฉันเห็นการอ้างสิทธิ์ของbytes.hexวิธีการbytes.decodeตัวแปลงสัญญาณและได้ลองฟังก์ชั่นอื่น ๆ ที่เป็นไปได้อย่างน่าประหลาดใจอย่างน้อยที่สุดโดยไม่มีประโยชน์ ฉันแค่อยากให้ไบต์เป็นเลขฐานสิบหก!


"ไม่มีประโยชน์"? คุณได้รับปัญหาหรือข้อผิดพลาดเฉพาะเรื่องใดบ้าง กรุณาแสดงรหัสและข้อผิดพลาด
S.Lott

ซ้ำซ้อนที่เป็นไปได้ของstackoverflow.com/questions/2340319/python-3-1-1-string-to-hex
Mu Mind

คำตอบ:


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'

ดูคำตอบนี้: สตริง Python 3.1.1 เป็น hex


8
ดีจัง. คุณสามารถแปลง hex เป็น bytes ได้โดยใช้ bytes.fromhex (hex_str) แต่คุณไม่สามารถแปลง bytes เป็น hex ได้โดยใช้ bytes.tohex () - อะไรคือเหตุผลของเรื่องนี้?
nagylzs

1
ฉันเดาว่าความสัมพันธ์ระหว่างไบต์และฐานสิบหกไม่ใช่คุณสมบัติของทั้งคู่ (ซึ่งไม่ตอบว่าทำไม fromhex อยู่ที่นั่น) ดูเหมือนว่ามันก็ไม่ได้เป็นเพียงการกำกับดูแล แต่สิ่งที่เป็นที่ถกเถียงกันมากกว่า: bugs.python.org/issue3532#msg70950 ถาม: เจ็บไหมที่จะใช้วิธี tohex ของอ็อบเจกต์ไบต์เพื่อทำงานนี้เช่นกัน? ตอบ: IMO ได้ มันซับซ้อนรหัสและดึงโฟกัสออกจากวิธีการที่เหมาะสมในการแปลงข้อมูล (เช่นฟังก์ชั่น - ไม่ใช่วิธีการ)
Mu Mind

3
สิ่งนี้ตอบคำถามได้จริงหรือ? มันไม่ได้กลับมาเป็นฐานสิบหกแต่str bytesฉันรู้ว่า OP ดูเหมือนจะมีความสุขกับคำตอบ แต่จะไม่ดีกว่าที่จะขยายคำตอบนี้เพื่อรวม.decode("ascii")การแปลงเป็นสตริง "
RubenLaguna

3
ผมคิดว่าหลาย ๆ คนลงจอดบนคำถามนี้ / bytesคำตอบที่กำลังมองหาวิธีการที่จะพิมพ์ออกมา หากคุณprint(b'666f6f')ได้รับbในงานพิมพ์ ถ้าคุณ.decode("ascii")คุณทำไม่ได้ แค่คิดว่าผู้ที่มีbytes(จริงไบนารีกับรายการ> 128 ไม่ใช่สตริง ascii) ต้องการพิมพ์ออกมาอย่างไร
RubenLaguna

5
@nagylzs: มี.hex()วิธีการใน Python 3.5+
jfs

43

Python มีตัวแปลงสัญญาณมาตรฐานแบบไบต์ต่อไบต์ที่ดำเนินการแปลงอย่างสะดวกสบาย (เช่นขนาด 7bits ascii), base64 (เหมาะกับตัวอักษรและตัวเลข), ฐานสิบหกหนี, การบีบอัดฐานสิบหก, gzip และ bz2 ใน Python 2 คุณสามารถทำได้:

b'foo'.encode('hex')

ใน Python 3 str.encode/ bytes.decodeมีไว้สำหรับ bytes <-> str conversion แต่คุณสามารถทำสิ่งนี้ได้ซึ่งใช้ได้กับ 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

6

วิธีการbinascii.hexlify()จะแปลงbytesเป็นbytesสตริง 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"ที่จะสามารถกลับมาbytesด้วยbinascii.unhexlify("0a160a04")ซึ่งจะช่วยให้กลับมาb'\n\x16\n\x04'


3

ตกลงคำตอบต่อไปนี้อยู่นอกเหนือขอบเขตเล็กน้อยหากคุณสนใจ Python 3 เท่านั้น แต่คำถามนี้เป็นคำตอบแรกของ Google แม้ว่าคุณจะไม่ได้ระบุเวอร์ชันของ Python ดังนั้นนี่เป็นวิธีที่ใช้ได้ทั้ง Python 2 และ Python 3 .

ฉันยังตีความคำถามที่เกี่ยวกับการแปลงไบต์เป็นstrประเภท: นั่นคือ bytes-y ใน Python 2 และ Unicode-y บน Python 3

ระบุว่าวิธีที่ดีที่สุดที่ฉันรู้คือ:

import six

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

การยืนยันต่อไปนี้จะเป็นจริงสำหรับ Python 2 หรือ Python 3 โดยสมมติว่าคุณไม่ได้เปิดใช้งานunicode_literalsPython 2 ในอนาคต:

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
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.