Code | Description |
---|---|
a | 以NUL字節(jié)填充字符串空白 |
A | 以SPACE(空格)填充字符串 |
h | 十六進制字符串,低位在前 |
H | 十六進制字符串,高位在前 |
c | 有符號字符 |
C | 無符號字符 |
s | 有符號短整型(16位,主機字節(jié)序) |
S | 無符號短整型(16位,主機字節(jié)序) |
n | 無符號短整型(16位,大端字節(jié)序) |
v | 無符號短整型(16位,小端字節(jié)序) |
i | 有符號整型(機器相關(guān)大小字節(jié)序) |
I | 無符號整型(機器相關(guān)大小字節(jié)序) |
l | 有符號長整型(32位,主機字節(jié)序) |
L | 無符號長整型(32位,主機字節(jié)序) |
N | 無符號長整型(32位,大端字節(jié)序) |
V | 無符號長整型(32位,小端字節(jié)序) |
q | 有符號長長整型(64位,主機字節(jié)序) |
Q | 無符號長長整型(64位,主機字節(jié)序) |
J | 無符號長長整型(64位,大端字節(jié)序) |
P | 無符號長長整型(64位,小端字節(jié)序) |
f | 單精度浮點型(機器相關(guān)大小) |
d | 雙精度浮點型(機器相關(guān)大小) |
x | NUL字節(jié) |
X | 回退一字節(jié) |
Z | 以NUL字節(jié)填充字符串空白(new in PHP 5.5) |
@ | NUL填充到絕對位置 |
這么多參數(shù)看下來,我第一次是真心懵逼了,大部分說明都很好理解,但是其中的主機、大端、小端等字節(jié)序是什么鬼呢?接下里的內(nèi)容比較枯燥,但必須理解才行,堅持吧。
字節(jié)序是什么?
就是字節(jié)的順序,說白了就是多字節(jié)數(shù)據(jù)的存放順序(一個字節(jié)顯然不需要順序)。
比如A和B分別對應(yīng)的二進制表示為0100 0001、0100 0010。對于儲存字符串AB,我們可以0100 0001 0100 0010也可以0100 0010 0100 0001,這個順序就是所謂的字節(jié)序。
高/低位字節(jié)
比如字符串AB,左高右低(我們正常的閱讀順序),A為高字節(jié),B為低字節(jié)
高/低地址
假設(shè)0x123456是按從高位到底位的順序儲存,內(nèi)存中是這樣存放的:
高地址 -> 低地址
12 -> 34 -> 56
大端字節(jié)序(網(wǎng)絡(luò)字節(jié)序)
大端就是將高位字節(jié)放到內(nèi)存的低地址端,低位字節(jié)放到高地址端。網(wǎng)絡(luò)傳輸中(比如TCP/IP)低地址端(高位字節(jié))放在流的開始,對于2個字節(jié)的字符串(AB),傳輸順序為:A(0-7bit)、B(8-15bit)。
那么小端字節(jié)序自然和大端相反。
主機字節(jié)序
表示當年機器的字節(jié)序(也就是網(wǎng)絡(luò)字節(jié)序是確定的,而主機字節(jié)序是依機器確定的),一般為小端字節(jié)序。
a和A(打包字符串,用NUL或者空格填充)
$string = pack('a6', 'china'); var_dump($string); //輸出結(jié)果: string(6) "china",最后一個字節(jié)是不可見的NUL echo ord($string[5]); //輸出結(jié)果: 0(ASCII碼中0對應(yīng)的就是nul) //A同理 $string = pack('A6', 'china'); var_dump($string); //輸出結(jié)果: string(6) "china ",最后一個字節(jié)是空格 echo ord($string[5]); //輸出結(jié)果: 32(ASCII碼中32對應(yīng)的就是空格)
附贈ASCII表一張(linux/unix下可以使用man ascii查看)
h和H
$string = pack('H3', 281); var_dump($string); //輸出結(jié)果: string(2) "(" for($i=0;$istrlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //輸出結(jié)果: 40 16
h和H需要特殊說明一下,它們是將對應(yīng)的參數(shù)看做十六進制字符然后打包。什么意思呢?比如上面的281,打包前會將281轉(zhuǎn)換為0x281,因為十六進制的一位對應(yīng)二進制的四位,上面的0x281只有1.5個字節(jié),后面會默認補0變成0x2810,0x28對應(yīng)的十進制為40((),0x10對應(yīng)的十進制為16(dle不可見字符),懂了吧?不懂可以給我留言。。
c和C
$string = pack('c3', 67, 68, -1); var_dump($string); //輸出:string(3) "CD�" for($i=0;$istrlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //輸出: 67 68 225
最后輸出本能應(yīng)該覺得是67 68 -1
ord獲取的是字符的ASCII碼(范圍0-255),這時-1(0000 0001)對應(yīng)的字符將以補碼的形式輸出也就是255(1111 1110 + 0000 0001 = 1111 1111)
整型相關(guān)
所有的整型類型使用方法完全一樣,主要注意它們的位和字節(jié)序就可以了,下面以L作為例子展示
$string = pack('L', 123456789); var_dump($string); //輸出:string(4) "�[" for($i=0;$istrlen($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是針對浮點數(shù)打包,至于為什么打包前是12345.123解包后是12345.123046875,這個和浮點數(shù)的儲存有關(guān)系,后面可以單開一個文章講解一下IEEE標準
x、X、Z、@
$string = pack('x'); //打包一個nul字符串 echo ord($string); //輸出: 0
關(guān)于X(大寫X),試了N次,沒搞明白怎么用,有清楚的童鞋可以給我留言,多謝。
$string = pack('Z2', 'abc5'); //其實就是將從Z后面的數(shù)字位置開始,全部設(shè)置為nul var_dump($string); //輸出:string(2) "a" for($i=0;$istrlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //輸出: 97 0
$string = pack('@4'); //我理解為填充N個nul var_dump($string); //輸出: string(4) "" for($i=0;$istrlen($string);$i++) { echo ord($string[$i]) . PHP_EOL; } //輸出: 0 0 0 0
unpack
array unpack ( string $format , string $data )
unpack的使用相當簡單,就是講pack打包的數(shù)據(jù)解包,打包的時候用的什么參數(shù),就用什么參數(shù)解包,具體使用懶得說了,列幾個小例子
$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) }
這兩個函數(shù)到底有啥用途
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。