Heim > Artikel > Backend-Entwicklung > Ausführliche Erläuterung der Verwendung von Pack und Unpack
Dieses Mal werde ich Ihnen die Verwendung von Ein- und Auspacken ausführlich erläutern. Was sind die Vorsichtsmaßnahmen bei der Verwendung von Ein- und Auspacken?
Es gibt zwei Funktionen Packen und Entpacken in PHP. Viele PHPer haben sie noch nie in tatsächlichen Projekten verwendet und wissen nicht einmal, wofür diese beiden Methoden verwendet werden. In diesem Artikel erfahren Sie, wofür sie verwendet werden.
pack
string pack ( string $format [, mixed $args [, mixed $... ]] )
Diese Funktion wird verwendet, um die entsprechenden Parameter ($args) in einen binären String zu packen.
Der erste Parameter $format hat die folgenden Optionen (es gibt viele optionale Parameter und einige häufig verwendete werden später erklärt):
Code | Beschreibung | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
a | Füllen Sie die Zeichenfolge mit NUL-Bytes|||||||||||||||||||||||||||||||||||||||||||||||||||||||
A | Füllen Sie die Zeichenfolge mit SPACE (Leerzeichen) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
Hexadezimale Zeichenfolge, niederwertiges Bit zuerst | |||||||||||||||||||||||||||||||||||||||||||||||||||||||
H | Hexadezimale Zeichenfolge, hochwertiges Bit zuerst | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
c | Zeichen mit Vorzeichen | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
C | Zeichen ohne Vorzeichen | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
s | Signed short (16 Bits, Host-Byte-Reihenfolge) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
S | Unsigned short (16-Bit, Host-Byte-Reihenfolge) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
n | Unsigned Short Integer-Typ (16 Bit, Big-Endian-Byte-Reihenfolge) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
v | Vorzeichenloser kurzer Ganzzahltyp (16 Bit, Little-Endian-Bytereihenfolge) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
i | Vorzeichenbehaftete Ganzzahl (maschinenabhängige Größen-Byte-Reihenfolge)|||||||||||||||||||||||||||||||||||||||||||||||||||||||
I | Vorzeichenlose Ganzzahl (maschinenabhängige Größen-Endianness) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
l | Signed Long (32-Bit, Host-Endianness) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
L | Unsigned Long Integer (32 Bit, Host-Endianness) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
N | Unsigned long (32-Bit, Big-Endian) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
V | Keine vorzeichenbehaftete lange Ganzzahl (32 Bit, Little Endian) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
q | vorzeichenbehaftete lange Ganzzahl (64 Bits, Host-Endian) ) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
Q | Unsigned Long Integer (64-Bit, Host-Byte-Reihenfolge) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
Unsigned Long Integer (64-Bit, Big-Endian) | |||||||||||||||||||||||||||||||||||||||||||||||||||||||
P | Unsigned Long Integer ( 64 Bit, Little Endian) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
f | Einfache Genauigkeit Gleitkomma
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||
d | Gleitkommatyp mit doppelter Genauigkeit (maschinenabhängige Größe) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
x | NUL-Bytes | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
X td> | Ein Byte zurücksetzen | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
Z | String-Lücken mit NUL-Bytes füllen (neu in PHP 5.5) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||
@ | NUL-Auffüllen zur absoluten Position |
这么多参数看下来,我第一次是真心懵逼了,大部分说明都很好理解,但是其中的主机、大端、小端等字节序是什么鬼呢?接下里的内容比较枯燥,但必须理解才行,坚持吧。
字节序是什么?
就是字节的顺序,说白了就是多字节数据的存放顺序(一个字节显然不需要顺序)。
比如A和B分别对应的二进制表示为0100 0001、0100 0010。对于储存字符串AB,我们可以0100 0001 0100 0010也可以0100 0010 0100 0001,这个顺序就是所谓的字节序。
高/低位字节
比如字符串AB,左高右低(我们正常的阅读顺序),A为高字节,B为低字节
高/低地址
假设0x123456是按从高位到底位的顺序储存,内存中是这样存放的:
高地址 -> 低地址
12 -> 34 -> 56
大端字节序(网络字节序)
大端就是将高位字节放到内存的低地址端,低位字节放到高地址端。网络传输中(比如TCP/IP)低地址端(高位字节)放在流的开始,对于2个字节的字符串(AB),传输顺序为:A(0-7bit)、B(8-15bit)。
那么小端字节序自然和大端相反。
主机字节序
表示当年机器的字节序(也就是网络字节序是确定的,而主机字节序是依机器确定的),一般为小端字节序。
a和A(打包字符串,用NUL或者空格填充)
$string = pack('a6', 'china'); var_dump($string); //输出结果: string(6) "china",最后一个字节是不可见的NUL echo ord($string[5]); //输出结果: 0(ASCII码中0对应的就是nul) //A同理 $string = pack('A6', 'china'); var_dump($string); //输出结果: string(6) "china ",最后一个字节是空格 echo ord($string[5]); //输出结果: 32(ASCII码中32对应的就是空格)
附赠ASCII表一张(linux/unix下可以使用man ascii查看)
h和H
$string = pack('H3', 281); var_dump($string); //输出结果: string(2) "(" for($i=0;$i<strlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //输出结果: 40 16
h和H需要特殊说明一下,它们是将对应的参数看做十六进制字符然后打包。什么意思呢?比如上面的281,打包前会将281转换为0x281,因为十六进制的一位对应二进制的四位,上面的0x281只有1.5个字节,后面会默认补0变成0x2810,0x28对应的十进制为40((),0x10对应的十进制为16(dle不可见字符),懂了吧?不懂可以给我留言。。
c和C
$string = pack('c3', 67, 68, -1); var_dump($string); //输出:string(3) "CD�" for($i=0;$i<strlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //输出: 67 68 225
最后输出本能应该觉得是67 68 -1
ord获取的是字符的ASCII码(范围0-255),这时-1(0000 0001)对应的字符将以补码的形式输出也就是255(1111 1110 + 0000 0001 = 1111 1111)
整型相关
所有的整型类型使用方法完全一样,主要注意它们的位和字节序就可以了,下面以L作为例子展示
$string = pack('L', 123456789); var_dump($string); //输出:string(4) "�[" for($i=0;$i<strlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //输出: 21 205 91 7
f和d
$string = pack('f', 12345.123); var_dump($string); //输出:string(4) "~�@F" var_dump(unpack('f', $string)); //这里提前用到了unpack,后面会讲解 //输出:float(12345.123046875)
f和d是针对浮点数打包,至于为什么打包前是12345.123解包后是12345.123046875,这个和浮点数的储存有关系,后面可以单开一个文章讲解一下IEEE标准
x、X、Z、@
$string = pack('x'); //打包一个nul字符串 echo ord($string); //输出: 0
关于X(大写X),试了N次,没搞明白怎么用,有清楚的童鞋可以给我留言,多谢。
$string = pack('Z2', 'abc5'); //其实就是将从Z后面的数字位置开始,全部设置为nul var_dump($string); //输出:string(2) "a" for($i=0;$i<strlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //输出: 97 0
$string = pack('@4'); //我理解为填充N个nul var_dump($string); //输出: string(4) "" for($i=0;$i<strlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //输出: 0 0 0 0
unpack
array unpack ( string $format , string $data )
unpack的使用相当简单,就是讲pack打包的数据解包,打包的时候用的什么参数,就用什么参数解包,具体使用懒得说了,列几个小例子
$string = pack('L4', 1, 2, 3, 4); var_dump(unpack('L4', $string)); //输出: array(4) { [1]=> int(1) [2]=> int(2) [3]=> int(3) [4]=> int(4) } $string = pack('L4', 1, 2, 3, 4); var_dump(unpack('Ll1/Ll2/Ll3/Ll4', $string)); //可以指定key,用/分割 //输出: array(4) { ["l1"]=> int(1) ["l2"]=> int(2) ["l3"]=> int(3) ["l4"]=> int(4) }
这两个函数到底有啥用途
数据通信(通过二进制格式与其它语言通信)
数据加密(如果不告诉第三方你的打包方式,对方解包的难度就相对很大)
节省空间(比如比较大的数字按字符串储存会浪费很多空间,打包成二进制格式才需要4位<32位数字>)
相信看了本文案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!
推荐阅读:
Post kann keine Formulardaten abrufen
Das obige ist der detaillierte Inhalt vonAusführliche Erläuterung der Verwendung von Pack und Unpack. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!