Python——struct模块简介

12 篇文章 1 订阅
11 篇文章 1 订阅

最近在做视频解封装的时候,遇到了struct这个模块的使用,查阅了一些文档,现在总结一下。

了解c/c++的人,一定会知道struct结构体在其中的作用:它定义了一种结构(可以理解为值类型),便于不同目的下使用时,封装新的数据类型。当传递一些复杂的数据结构的时候,需要有一种机制将某些特定的结构体类型打包成二进制流的字符串然后再网络传输,而接收端也应该可以通过某种机制进行解包还原出原始的结构体数据。python中的struct模块就提供了这样的机制,该模块的主要作用就是对Python值类型(如string, int, float等)与用Python string表示的C struct类型间的转化(This module performs conversions between Python values and C structs represented as Python strings.)。

在struct模块中,最重要也是最为常用的三个函数分别是pack(), unpack(), calcsize()。下面来举例介绍一下它们的用法:

pack unpack calcsize等函数的使用

  • struct.pack(fmt, v1, v2, …)
    返回包含值v1,v2,…的字符串,并将其根据给定的参数fmt进行pack。这里,参数必须和值对位匹配。
    比如pack('2h',1,2,3)就会报错,因为format只有2个short型,不匹配。

  • struct.pack_into(fmt, buffer, offset, v1, v2, …)
    将v1,v2,…按指定的格式(fmt)进行Pack,将Packed字节写入到可写的buffer字符串中(需要指定offset——可理解为位置,注意offset不能省略)。

  • struct.unpack(fmt, string)
    根据指定格式Unpack字符串。返回的结果是元组(tuple)形式。注意,这里字符串必须包含足够的数据以确保len(string)与calcsize(fmt)相等。(The string must contain exactly the amount of data required by the format (len(string) must equal calcsize(fmt))。

  • struct.unpack_from(fmt, buffer[, offset=0])
    是pack_into的反向操作,参看下面的例子。

  • struct.calcsize(fmt)
    返回格式对应的C类型字节之和。

unpack和pack的例子

>>> from struct import *
>>> import binascii
>>> pack('hhl', 1, 2, 3)
'\x00\x01\x00\x02\x00\x00\x00\x03'
>>> unpack('hhl', '\x00\x01\x00\x02\x00\x00\x00\x03')
(1, 2, 3)
>>> calcsize('hhl') # 'hhl' 与 '2hl'是一样的。
8
>>> print binascii.hexlify('\x00\x01\x00\x02\x00\x00\x00\x03')
0001000200000003

这个例子是:先将数字1,2,3打包为一个pack,然后再用unpack解出来。calcsize()是计算它对应的长度。具体见下图:
2个h为2*2=4,1个l为4,所以和为8.

这里写图片描述

字节顺序、大小和对齐

默认情况下,Pack后的字节顺序默认上是由操作系统的决定的。struct模块提供了自定义字节顺序的功能,可以指定大端存储、小端存储等特定的字节顺序(通过在format的第一个位置处加上字符,默认是@
在format字符串前面加上特定的符号即可以表示不同的字节顺序存储方式,例如采用小端存储
s = struct.Struct(‘〈hhl’)就可以了。Python官方提供了相应的对照列表:

这里写图片描述

使用pack_into和unpack_from方法

使用二进制pack数据的场景大部分都是对性能要求比较高的使用环境。而在上面提到的pack方法都是对输入数据进行操作后重新创建了一个内存空间用于返回,也就是说我们每次pack都会在内存中分配出相应的内存资源,这有时是一种很大的性能浪费。struct模块还提供了pack_into() 和 unpack_from()的方法用来解决这样的问题,也就是对一个已经提前分配好的buffer进行字节的填充,而不会每次都产生一个新对象对字节进行存储。

对比使用pack方法,pack_into 一直是在对buffer对象进行操作,没有产生多余的内存浪费。另外需要注意的一点是,pack_into和unpack_from方法均是对string buffer对象进行操作,并提供了offset参数,用户可以通过指定相应的offset,使相应的处理变得更加灵活。例如,我们可以把多个对象pack到一个buffer里面,然后通过指定不同的offset进行unpack(这里的offset就是各个struct的size):

import ctypes
from struct import *
import binascii

values1 = ('abc', 1000, 8000)
values2 = ('defg', 101)
values3 = (8888, 1.6)
s1 = Struct('3sLL')
s2 = Struct('4sI')
s3 = Struct('Lf')
# 创建一个用于缓存内容的buffer
buffer = ctypes.create_string_buffer(s1.size + s2.size + s3.size)

print type(buffer)
# <class 'ctypes.c_char_Array_28'>

print 'Before :', binascii.hexlify(buffer)

s1.pack_into(buffer, 0, *values1)
s2.pack_into(buffer, s1.size, *values2)
s3.pack_into(buffer, s1.size + s2.size, *values3)

print 'After pack:', binascii.hexlify(buffer)

print s1.unpack_from(buffer, 0)
print s2.unpack_from(buffer, s1.size)
print s3.unpack_from(buffer, s1.size+s2.size)

输出:
<class 'ctypes.c_char_Array_28'>
Before : 00000000000000000000000000000000000000000000000000000000
After pack: 61626300e8030000401f00006465666765000000b8220000cdcccc3f
('abc', 1000, 8000)
('defg', 101)
(8888, 1.600000023841858)

值得注意,这里float的精度发生了改变(从1.6变为1.60000…),这是由一些比如操作系统等客观因素所决定的。

格式字符的顺序可能对大小有影响,因为满足对齐需求所需的填充是不同的

import ctypes
from struct import *
import binascii

pack('ci', '*', 0x12131415)
# '*\x00\x00\x00\x12\x13\x14\x15'
pack('ic', 0x12131415, '*')
# '\x12\x13\x14\x15*'
print calcsize('ci')
# 8
print calcsize('ic')
# 5
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: struct模块的unpack函数用于将二进制数据解包成Python对象。它的语法为:struct.unpack(format, buffer)。其中,format是解包时使用的格式字符串,buffer是要解包的二进制数据。解包后的结果是一个元组,其中每个元素对应一个解包后的值。 ### 回答2: Python中的struct模块是用于处理二进制数据的一个强大的模块,它提供了一种轻松快速地在Python程序中处理各种二进制数据格式的方法。 在struct模块中,unpack()函数是一个十分重要的函数,它可以将二进制数据解析并转换为Python中的变量类型。 unpack()函数的语法如下: struct.unpack(format, buffer) 其中,format参数是一个字符串,表示待解析的二进制数据的格式,它是由一些字母和数字组成的。buffer参数则是一个字节数组或者字节串,表示需要解析的二进制数据。 在format字符串中,不同的字母和数字表示不同的数据类型和大小。例如,字母“i”表示一个有符号的整数,数字“4”表示该整数占用的字节数。 在使用unpack()函数时,需要根据待解析的二进制数据的格式指定相应的format字符串,然后将待解析的二进制数据作为buffer参数传入。unpack()函数将会返回一个元组,其中包含了按照format字符串解析后得到的各个值。这些值的类型和顺序与format字符串中的说明相对应。 需要注意的是,如果待解析的二进制数据的实际格式与指定的format字符串不一致,那么unpack()函数将会抛出一个struct.error异常,因此,在使用unpack()函数时一定要确保指定的format字符串与实际的二进制数据格式相匹配。 总之,使用Python中的struct模块的unpack()函数可以方便地将二进制数据解析为Python中的变量类型,从而方便地进行数据处理和操作。 ### 回答3: Python中的struct模块提供解析字节串(bytes)和打包成字节串的函数,是进行二进制数据处理的重要模块之一。其中,unpack函数可以将字节串解析成Python对象,用于读取和解析来自二进制文件、网络连接等的数据。 unpack函数的基本语法为: struct.unpack(format, buffer) 其中,format是解析字节串的格式码,buffer是需要解析的字节串。 格式码(format)包括一个或多个类型字符,用来指定解析字节串的数据类型和顺序。常用的数据类型包括整型、浮点型、字符型、字节型等,例如: ‘i’表示整型; ‘f’表示单精度浮点型; ‘s’表示字符串(需指定长度); ‘b’表示有符号字节型; ‘H’表示无符号短整型等。 在解析字节串时,需要使用与打包时相同的格式码,按照顺序解析出各个数据成员,并将它们转换成对应的Python对象。例如,以下是一个解析包含一个整型和一个字符型的字节串的示例代码: import struct data = b'\x01A' i, c = struct.unpack('i1s', data) print(i, c) 解析结果为: 1 b'A' 在解析时,将‘i’和‘1s’格式码传递给unpack函数,程序按照解析顺序依次将字节串中的两个数据成员解析成Python对象,返回一个包含两个Python对象的元组。在本例中,字节串中第一个字节表示整型1,第二个字节表示字符‘A’,故输出结果为1和‘A’。 需要注意的是,在解析字节串时,需要按照打包时指定的顺序和格式码进行解析,否则会导致数据解析错误。因此,在数据传输和存储时,需要使用相同的打包和解析格式码,保证数据的正确传输和解析。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值