2

我实际上想知道 python3 的 struct.pack 行为,但也许我错过了一些东西。

我正在通过 UDP 提交一个 .jpg 文件。顺便说一句:当我尝试一次提交整个文件 (~200kB) 时, socket.sendTo()and函数会抛出“Python IOError: [Errno 90] Message too long”。sendAll()所以我以 1024 字节为单位提交文件。没有问题,我只是想知道为什么我在 python 文档中没有发现这个大小限制。

无论如何,我的主要问题是:我需要 struct.pack 在每个片段的开头放置一些信息-> 2 个固定大小的字符串。

但是当我这样做的时候

chunk = struct.pack("!3c4cI", bytes("JPG", "utf-8"), bytes(dev_id, "utf-8"), i)

它是“struct.error:包装预期的 8 个包装物品(得到 3 个)”

所以我得走了

chunk = struct.pack("!3c4ci", b"J", b"P", b"G", 
    bytes(dev_id[0:1], "utf-8"),
    bytes(dev_id[1:2], "utf-8"),
    bytes(dev_id[2:3], "utf-8"),
    bytes(dev_id[3:4], "utf-8"), i)

让它工作。这是为什么!?

4

1 回答 1

3

struct.pack要求每个项目作为单独的参数传递。char此外,Python 对和进行了区分byte,即使在 C 中它们是同义词,这就是为什么你需要为 0 设置一个字节bytes的值c,而不是 0 .. 255 范围内的整数。

但是,struct也支持s格式说明符,其中s代表给定长度的字符串:

>>> dev_id, i = 'R2D2', 42
>>> struct.pack("!3s4sI", b"JPG", dev_id.encode(), i)
b'JPGR2D2\x00\x00\x00*'

或者,如果您至少使用 Python 3.5,那么感谢PEP 448 -- Additional Unpacking Generalizations您可以将B(unsigned byte) 格式与 splat 运算符结合使用,*如下所示:

>>> struct.pack("!3B4BI", *b"JPG", *dev_id.encode(), i)

这里*所做的是将给定bytes值中的每个字节作为 0 ... 255 范围内的整数解压缩到单独的参数中;如果dev_id.encode()结果为 4 个 UTF-8 字节,则总共 8 个参数将传递给struct.pack. 与 不同的是cB接受单字节值作为整数。


PS请注意,我b'JPG'直接使用而不是调用bytes('JPG', 'UTF-8'),并且同样调用了默认.encode()编码的字符串,UTF-8以获得更短的代码。

于 2016-10-03T19:43:29.660 回答