在 Python 中从 int 中提取位域

now*_*wox 5 python bit bit-fields

我有一个数字0x5423,我想提取 4 个值:

a = 0x5   # 15 downto 12
b = 0x42  # 11 downto 3
c = 0x3   # 3 downto 2
d = 0x00  # 1 downto 0
Run Code Online (Sandbox Code Playgroud)

我发现模块位串看起来很棒。不幸的是,由于未知的原因,这些位是从右侧开始编号的。

这很糟糕,因为如果添加一些像0xA5423我的提取这样的高位将不再起作用:

field = bitstrings.BitArray('0x5423')
a = field[0:4].uint
b = field[4:12].uint
c = field[12:14].uint
d = field[14:16].uint
Run Code Online (Sandbox Code Playgroud)

如何在不进行复杂算术操作的情况下正确提取位域,例如:

b = (a >> 4) & 0xFF
Run Code Online (Sandbox Code Playgroud)

理想情况下我会:

b = field.range(11, 4)
Run Code Online (Sandbox Code Playgroud)

fal*_*tru 2

在传递给之前将字符串转换为0x####格式bitstring.BitArray

>>> n = '0xA5423'
>>> n = '0x{:04x}'.format(int(n, 16) & 0xffff)  # => '0x5423'
>>> field = bitstring.BitArray(n)
>>> field[0:4].uint
5
>>> field[4:12].uint  # 0x42 == 66
66
>>> field[12:14].uint
0
>>> field[14:16].uint
3
Run Code Online (Sandbox Code Playgroud)

更新另一个不依赖于 的解决方案bitstring,并从左开始计数(根据OP):

将数字转换为二进制格式:

>>> n = '0xA5423'
>>> n = format(int(n, 16), '016b')[::-1]  # reversed
>>> n
'11000100001010100101'
>>> int(n[0:2][::-1], 2)  # need to reverse again to get proper value
3
>>> int(n[2:4][::-1], 2)
0
>>> int(n[4:12][::-1], 2)
66
>>> int(n[12:16][::-1], 2)
5
Run Code Online (Sandbox Code Playgroud)