转换十六进制

2024-06-16 10:11:53 发布

您现在位置:Python中文网/ 问答频道 /正文

如何在Python中将以下十六进制字符串转换为浮点(单精度32位)?

"41973333" -> 1.88999996185302734375E1

"41995C29" -> 1.91700000762939453125E1

"470FC614" -> 3.6806078125E4

Tags: 字符串中将浮点单精度e4e1
3条回答

我猜这个问题与this one有关,您使用的是4个字节而不是8个十六进制数字。

"\x41\x91\x33\x33"是一个4字节的字符串,即使它看起来像16

>>> len("\x41\x91\x33\x33")
4
>>> import struct  
>>> struct.unpack(">fff","\x41\x97\x33\x33\x41\x99\x5C\x29\x47\x0F\xC6\x14")
(18.899999618530273, 19.170000076293945, 36806.078125)

如果确实需要处理十六进制字符串而不是实际字节,可以使用struct.pack来转换它,如下所示

>>> for hx in ["41973333","41995C29","470FC614"]:
...     print(struct.unpack(">f",struct.pack(">i",int(hx,16)))[0])
... 
18.8999996185
19.1700000763
36806.078125

我建议使用the ctypes module,它基本上允许您处理低级数据类型。在你的情况下你可以说

from ctypes import *

def convert(s):
    i = int(s, 16)                   # convert from hex to a Python int
    cp = pointer(c_int(i))           # make this into a c integer
    fp = cast(cp, POINTER(c_float))  # cast the int pointer to a float pointer
    return fp.contents.value         # dereference the pointer, get the float

print convert("41973333")    # returns 1.88999996185302734375E1

print convert("41995C29")    # returns 1.91700000762939453125E1

print convert("470FC614")    # returns 3.6806078125E4

我相信ctypes模块在这里是有意义的,因为您实际上是在问如何执行低级位转换。基本上,您的问题是,我如何告诉Python获取一些数据并将这些数据解释为那些完全相同的位是不同的数据类型?

在C语言中,如果你有一个int,想把它的位解释成一个浮点,你可以做大致相同的事情,取一个指针,然后对其进行转换和解引用:

int i = 0x41973333;
float f = *((float*)&i);

在我的示例中,使用ctypes库的Python代码正是这样做的。

>>> import struct
>>> struct.unpack('!f', '41973333'.decode('hex'))[0]
18.899999618530273
>>> struct.unpack('!f', '41995C29'.decode('hex'))[0]
19.170000076293945
>>> struct.unpack('!f', '470FC614'.decode('hex'))[0]
36806.078125

更新:请参阅Python3中的注释。

相关问题 更多 >