pack函数说明:本函数用来将资料压缩打包到位的字符串之中。
首先 pack的 格式 为 pack('最原始数据使用的格式,'原始数据'');
pack('a1','h')或者pack('a*','h');
每个字母(比如上面的a)后面都可以跟着一个数字,表示count(计数)来解释成某种形式的重复计数或者长度(换算后字符长度发生了改变),具体情况取决于格式。除了a,A,b,B,h,H,P,和 Z 之外,所有格式的count 都是重复次数(也就是说 上面的这几个字符后面的“计数”指的是获取字符串的长度),ack('a*','h')字母(比如上面的a)后面都可以跟着一个数字,表示count(计数),解释成某种形式的重复计数或者长度,具体情况取决于格式。除了a,A,b,B,h,H,P,和 Z 之外,所有格式的count 都是重复次数(也就是说 上面的这几个字符后面的“计数”指的是获取字符串的长度),因此 pack 从 LIST 里吃进那么多数值(除了第一个参数,剩下的参数可以理解成都在一个列表中)。如果 count 是一个 *表示剩下的所有东西。【pack('a*',100,20) :只取第一个参数的所有字符 ,pack('C*',100,101,102):把这三个参数都换算,pack('Cf'',224,3.14,100) :C:换算244,f:换算剩下所有的参数】。
理解:pack 返回的数据 都用通过 ASCII码换算过后的,如果单纯理解ASCII换算的话 ,可以用一个php函数 ord():返回字符串第一个字符的 ASCII 值。
比如 ord(‘h’)和ord(‘hello’)都是返回一个十进制数:114【占用三个字节】,而pack('a*','h')也返回114, 但是在ord的基础上对字节进行了 增加或者减少(也就是对位的增加减少)当前这个列子上没有增加也没减少 比如pack('C',100) 返回 一个字节为:d。
上面的"a" 指的是转成什么字符,“*”指的是剩下的所有东西,如果改成‘a1’的话 ,就是获取原始数据的第一个字符转成a格式的字符。
下面是pack所有的格式(网上全部都是列出来的,虽然一看见这么多 觉着恶心):
a 一个填充空的字节串
A 一个填充空格的字节串 b 一个位串,在每个字节里位的顺序都是升序 B 一个位串,在每个字节里位的顺序都是降序 c 一个有符号 char(8位整数)值 C 一个无符号 char(8位整数)值;关于 Unicode 参阅 U d 本机格式的双精度浮点数 f 本机格式的单精度浮点数 h 一个十六进制串,低四位在前 H 一个十六进制串,高四位在前 i 一个有符号整数值,本机格式 I 一个无符号整数值,本机格式 l 一个有符号长整形,总是 32 位 L 一个无符号长整形,总是 32 位 n 一个 16位短整形,“网络”字节序(大头在前) N 一个 32 位短整形,“网络”字节序(大头在前) p 一个指向空结尾的字串的指针 P 一个指向定长字串的指针 q 一个有符号四倍(64位整数)值 Q 一个无符号四倍(64位整数)值 s 一个有符号短整数值,总是 16 位 S 一个无符号短整数值,总是 16 位 u 一个无编码的字串 U 一个 Unicode 字符数字 v 一个“VAX”字节序(小头在前)的 16 位短整数 V 一个“VAX”字节序(小头在前)的 32 位短整数 w 一个 BER 压缩的整数 x 一个空字节(向前忽略一个字节) X 备份一个字节 Z 一个空结束的(和空填充的)字节串 @ 用空字节填充绝对位置
通过下面的代码 可以看出他们的不同:
//A字符 $str=(pack("A*", "中国")); echo $str,"=",strlen($str),"字节\n"; getAscill($str); //H字符 $str=(pack("H*", "fffe")); echo $str,"=",strlen($str),"字节\n"; getAscill($str); //C字符 $str=(pack("C*", "55","56","57")); echo $str,"=",strlen($str),"字节\n"; getAscill($str); //i字符 短整形 32位 4个字节 64位8个字节 $str=(pack("i", "100")); echo $str,"=",strlen($str),"字节\n"; getAscill($str); //s字符 短整形 2个字节 $str=(pack("s", "100")); echo $str,"=",strlen($str),"字节\n"; getAscill($str); //l字符 长整形 4个字节 $str=(pack("l", "100")); echo $str,"=",strlen($str),"字节\n"; getAscill($str); //f字符 单精度浮点 4个字节 $str=(pack("f", "100")); echo $str,"=",strlen($str),"字节\n"; getAscill($str); //d字符 双精度浮点 8个字节 $str=(pack("d", "100")); echo $str,"=",strlen($str),"字节\n"; getAscill($str); function getAscill($str) { $arr=str_split($str); foreach ($arr as $v) { echo $v,"=",ord($v),"\n"; } echo "=============\r\n\r\n"; }
/*pack作用:节省空间、加密格式 */
$code = array(? "username" => array("A20", "张三adfb12"), "pass" => array("A10", "asdf*#1"), "age" => array("C", "23"), // 转成一个字节? 节省了空间 "birthday" => array("I", "19900101"), // 转成4个字节,想象一下 如果第二个参数长度一直增大比如‘199001010001256325’ ,到那时就更可以看出节省的空间了(因为不论多大 都是占用4个字节) "email" => array("A50", "zhangsan@163.com")); //进行了加密$stream = join("\0", packByArr($code));echo $stream, strlen($stream);file_put_contents("c:/1.txt", $stream); // 将流保存起来便于下面读取?function packByArr($arr)?{ $atArr = array(); foreach ($arr as $k => $v) { $atArr[] = pack($v[0], $v[1]); } return $atArr;} function getAscill($str){ $arr = str_split($str); foreach ($arr as $v) { echo $v, "=", ord($v), "\n"; } }
Unpack解包
解包需要按照,打包方式读取,该读取多长,该用什么类型读取,必须与打包规定一样。$code = array(? "username" => array("A20"), "pass" => array("A10"), "age" => array("C"), "birthday" => array("I"), "email" => array("A50"));$stream = file_get_contents("c:/1.txt");var_dump(packByArr($stream, $code));function packByArr($str, $code){ $Arr = explode("\0", $str); $atArr = array(); $i = 0; foreach ($code as $k => $v) { $atArr[$k] = unpack($v[0], $Arr[$i]); $i++; } return $atArr;}
$token="e3bc81ef8fa211a9edaaac2e0f7132d8";$id=123;$new_id=134343;$model="image";$msg = pack("NA40NN", strlen($token), $token, $id,$new_id);print $msg;$rs = unpack("NLength/A40token/Nid/Ncmsaid/A10model" , $msg);if($rs==NULL){ $rs = unpack("NLength/A40token/Nid/Ncmsaid" , $msg);}print_r($rs);print $rs['id'];