Python:struct模块的pack、unpack
mport struct
pack、unpack、pack_into、unpack_from

1 # ref: http://blog.csdn<a href="http://lib.csdn.net/base/dotnet" class='replace_word' title=".NET知识库" target='_blank' style='color:#df3434; font-weight:bold;'>.NET</a>/JGood/archive/2009/06/22/4290158.aspx
2
3 import struct
4
5 #pack - unpack
6 print
7 print '===== pack - unpack ====='
8
9 str = struct.pack("ii", 20, 400)
10 print 'str:', str
11 print 'len(str):', len(str) # len(str): 8
12
13 a1, a2 = struct.unpack("ii", str)
14 print "a1:", a1 # a1: 20
15 print "a2:", a2 # a2: 400
16
17 print 'struct.calcsize:', struct.calcsize("ii") # struct.calcsize: 8
18
19
20 #unpack
21 print
22 print '===== unpack ====='
23
24 string = 'test astring'
25 format = '5s 4x 3s'
26 print struct.unpack(format, string) # ('test ', 'ing')
27
28 string = 'he is not very happy'
29 format = '2s 1x 2s 5x 4s 1x 5s'
30 print struct.unpack(format, string) # ('he', 'is', 'very', 'happy')
31
32
33 #pack
34 print
35 print '===== pack ====='
36
37 a = 20
38 b = 400
39
40 str = struct.pack("ii", a, b)
41 print 'length:', len(str) #length: 8
42 print str
43 print repr(str) # '/x14/x00/x00/x00/x90/x01/x00/x00'
44
45
46 #pack_into - unpack_from
47 print
48 print '===== pack_into - unpack_from ====='
49 from ctypes import create_string_buffer
50
51 buf = create_string_buffer(12)
52 print repr(buf.raw)
53
54 struct.pack_into("iii", buf, 0, 1, 2, -1)
55 print repr(buf.raw)
56
57 print struct.unpack_from("iii", buf, 0)

运行结果:
[work@db-testing-com06-vm3.db01.baidu.com Python]$ python struct_pack.py
===== pack - unpack =====
str: ?
len(str): 8
a1: 20
a2: 400
struct.calcsize: 8
===== unpack =====
('test ', 'ing')
('he', 'is', 'very', 'happy')
===== pack =====
length: 8
?
'/x14/x00/x00/x00/x90/x01/x00/x00'
===== pack_into - unpack_from =====
'/x00/x00/x00/x00/x00/x00/x00/x00/x00/x00/x00/x00'
'/x01/x00/x00/x00/x02/x00/x00/x00/xff/xff/xff/xff'
(1, 2, -1)
Python是一门非常简洁的语言,对于数据类型的表示,不像其他语言预定义了许多类型(如:在C#中,光整型就定义了8种)
它只定义了六种基本类型:字符串,整数,浮点数,元组(set),列表(array),字典(key/value)
通过这六种数据类型,我们可以完成大部分工作。但当Python需要通过网络与其他的平台进行交互的时候,必须考虑到将这些数据类型与其他平台或语言之间的类型进行互相转换问题。打个比方:C++写的客户端发送一个int型(4字节)变量的数据到Python写的服务器,Python接收到表示这个整数的4个字节数据,怎么解析成Python认识的整数呢? Python的标准模块struct就用来解决这个问题。
struct模块的内容不多,也不是太难,下面对其中最常用的方法进行介绍:
1、 struct.pack
struct.pack用于将Python的值根据格式符,转换为字符串(因为Python中没有字节(Byte)类型,可以把这里的字符串理解为字节流,或字节数组)。其函数原型为:struct.pack(fmt, v1, v2, ...),参数fmt是格式字符串,关于格式字符串的相关信息在下面有所介绍。v1, v2, ...表示要转换的python值。
2、 struct.unpack
struct.unpack做的工作刚好与struct.pack相反,用于将字节流转换成python数据类型。它的函数原型为:struct.unpack(fmt, string),该函数返回一个元组。
下面的例子将两个整数转换为字符串(字节流):
import struct a =
b =
s = struct.pack('ii', a, b)
print(s, type(s))
#输出:b'\x14\x00\x00\x00\x90\x01\x00\x00' <class 'bytes'>
print('length: ', len(s))
#输出:length:
s2 = struct.unpack('ii', s)
print(s2)
#输出:(, ) s2 = struct.unpack('ii', s)
#报错:unpack requires a buffer of bytes
#==>解压需要一个4字节的缓冲区,也就是说'ii'表示8个字节的缓冲
#格式符"i"表示转换为int,'ii'表示有两个int变量。
#进行转换后的结果长度为8个字节(int类型占用4个字节,两个int为8个字节)
可以使用python的内置函数repr来获取可识别的字符串,其中十六进制的0x00000014, 0x00001009分别表示20和400。
3、 struct.calcsize
struct.calcsize用于计算格式字符串所对应的结果的长度,如:struct.calcsize('ii'),返回8。因为两个int类型所占用的长度是8个字节。

1 import struct
2 print "len: ", struct.calcsize('i') # len: 4
3 print "len: ", struct.calcsize('ii') # len: 8
4 print "len: ", struct.calcsize('f') # len: 4
5 print "len: ", struct.calcsize('ff') # len: 8
6 print "len: ", struct.calcsize('s') # len: 1
7 print "len: ", struct.calcsize('ss') # len: 2
8 print "len: ", struct.calcsize('d') # len: 8
9 print "len: ", struct.calcsize('dd') # len: 16

4、 struct.pack_into、 struct.unpack_from
这两个函数在Python手册中有所介绍,但没有给出如何使用的例子。其实它们在实际应用中用的并不多。Google了很久,才找到一个例子,贴出来共享一下:

1 #!/usr/bin/env python
2 #encoding: utf8
3
4 import sys
5 reload(sys)
6 sys.setdefaultencoding("utf-8")
7
8 import struct
9 from ctypes import create_string_buffer
10
11 buf = create_string_buffer(12)
12 print repr(buf.raw) # '\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'
13
14 struct.pack_into("iii", buf, 0, 1, 2, -1)
15 print repr(buf.raw) # '\x01\x00\x00\x00\x02\x00\x00\x00\xff\xff\xff\xff'
16
17 print struct.unpack_from("iii", buf, 0) # (1, 2, -1)

struct 类型表
| Format | C Type | Python type | Standard size | Notes |
|---|---|---|---|---|
| x | pad byte | no value | ||
| c | char | string of length 1 | 1 | |
| b | signed char | integer | 1 | (3) |
| B | unsigned char | integer | 1 | (3) |
| ? | _Bool | bool | 1 | (1) |
| h | short | integer | 2 | (3) |
| H | unsigned short | integer | 2 | (3) |
| i | int | integer | 4 | (3) |
| I | unsigned int | integer | 4 | (3) |
| l | long | integer | 4 | (3) |
| L | unsigned long | integer | 4 | (3) |
| q | long long | integer | 8 | (2), (3) |
| Q | unsigned long long | integer | 8 | (2), (3) |
| f | float | float | 4 | (4) |
| d | double | float | 8 | (4) |
| s | char[] | string | 1 | |
| p | char[] | string | ||
| P | void * | integer | (5), (3) |
Python:struct模块的pack、unpack的更多相关文章
- python struct中的pack unpack
python struct中的pack unpack pytyon tuple元组 print struct.unpack("!ihb", buffer) 结果为7 //pyth ...
- Python学习——struct模块的pack、unpack示例
he struct module includes functions for converting between strings of bytes and native Python data t ...
- Python struct模块
有的时候需要用python处理二进制数据,比如,存取文件,socket操作时.这时候,可以使用python的struct模块来完成.可以用 struct来处理c语言中的结构体. struct模块中最重 ...
- python struct模块的使用
struct模块中的函数 函数 return explain pack(fmt,v1,v2…) string 按照给定的格式(fmt),把数据转换成字符串(字节流),并将该字符串返回. pack_in ...
- Python使用struct处理二进制(pack和unpack用法)
转载自:http://www.cnblogs.com/gala/archive/2011/09/22/2184801.html 这篇文章写的很好,所以无耻的转了.. 有的时候需要用python处理二进 ...
- c语言write与python的struct模块交互
以下讲的都是用二进制形式打开文件.网上有很多struct模块的文章,下面是我做的小实验. 1.对于c里面的fwrite写入一个单字节,写的就是它的二进制.如3,写入文件就是二进制0x03,它并不是3的 ...
- python中struct模块及packet和unpacket
转自:http://www.cnblogs.com/gala/archive/2011/09/22/2184801.html 我们知道python只定义了6种数据类型,字符串,整数,浮点数,列表,元组 ...
- 浅析Python中的struct模块
最近在学习python网络编程这一块,在写简单的socket通信代码时,遇到了struct这个模块的使用,当时不太清楚这到底有和作用,后来查阅了相关资料大概了解了,在这里做一下简单的总结. 了解c语言 ...
- 【转】浅析Python中的struct模块
[转]浅析Python中的struct模块 最近在学习python网络编程这一块,在写简单的socket通信代码时,遇到了struct这个模块的使用,当时不太清楚这到底有和作用,后来查阅了相关资料大概 ...
随机推荐
- api 爬虫 避免相同 input 在信息未更新 情况下 重复请求重复
限
- Django redis2 列表 和其他操作
列表的操作 List操作,redis中的List在在内存中按照一个name对应一个List来存储.如图: lpush插值至列表最左边 lpush(name,values) # 在name对应的list ...
- 使用parted 对大容量盘进行分区
MBR分区表:(MBR含义:主引导记录) 所支持的最大卷:2T (T; terabytes,1TB=1024GB) 对分区的设限:最多4个主分区或3个主分区加一个扩展分区. GPT分区表:(GPT含义 ...
- jQuery实现复选框全选/所有取消/反选/获得选择的值
<!DOCTYPE html> <html> <head> <script type="text/javascript" src=&quo ...
- 2014阿里实习生面试题——MySQL如何实现索引的
这是2014阿里实习生北京站二面的一道试题: 在MySQL中,索引属于存储引擎级别的概念,不同存储引擎对索引的实现方式是不同的,比如MyISAM和InnoDB存储引擎. MyISAM索引实现: MyI ...
- CentOS6安装DaoCloud加速器
天朝的网,你又不是不懂.我最爱的红杏最近也用不了了.FUCK GFW. 在这,我们使用DaoCloud的加速器,打开网址 https://dashboard.daocloud.io/mirror 找到 ...
- R语言set.seed()函数介绍
set.seed(),该命令的作用是设定生成随机数的种子,种子是为了让结果具有重复性.如果不设定种子,生成的随机数无法重现.这个函数的主要目的,是让你的模拟能够可重复出现,因为很多时候我们需要取随机数 ...
- 说说JavaScript 中的new吧
在其他语言中,new操作符都是用来实例化创建一个对象的,JavaScript 中同样如此,但是它又有一些不同.为了说清楚这个问题我们先来看一下JavaScript 中的类.原型.原型链.继承这些概念吧 ...
- web框架详解之tornado 一 模板语言以及框架本质
一.概要 Tornado 是 FriendFeed 使用的可扩展的非阻塞式 web 服务器及其相关工具的开源版本.这个 Web 框架看起来有些像web.py 或者 Google 的 webapp,不过 ...
- vim打开多个文件方式及操作
格式如下: #vim file*.txt 或者 #vim file file2 file3 查看当前编程的是那个文件,在冒号命令行下 :args 命令,类似:file [file2],以中括号里面为当 ...