park,unpark,ord這3個函數,在我們工作中,用到它們的估計不多。 我在最近一個工作中,因為通訊需要用到二進位流,然後介面用php接收。當時在處理時候,查閱不少資料。因為它們使用確實比較少,也很少朋友工作中會用到它們。 在工作中,我也逐漸瞭解到park,unpark,ord對於二進位位元組處理的強大。 下面我逐一介紹它們。
park,unpark,ord函數使用介紹
park函數說明:本函數用來將資料壓縮打包到位的字串之中。
文法:pack(format,args+)
參數 |
描述 |
format |
必需。規定在封裝資料時所使用的格式。 |
args+ |
可選。規定被封裝的一個或多個參數。 |
字元 |
說明 |
a |
將字串空白以 NULL 字元填滿 |
A |
將字串空白以 SPACE 字元 (空格) 填滿 |
h |
十六進位字串,低位在前 |
H |
十六進位字串,高位在前 |
c |
有號字元 |
C |
無號字元 |
s |
有號短整數 (十六位,依電腦的位順序) |
S |
無號短整數 (十六位,依電腦的位順序) |
n |
無號短整數 (十六位, 高位在後的順序) |
v |
無號短整數 (十六位, 低位在後的順序) |
i |
有號整數 (依電腦的順序及範圍) |
I |
無號整數 (依電腦的順序及範圍) |
l |
有號長整數 (卅二位,依電腦的位順序) |
L |
無號長整數 (卅二位,依電腦的位順序) |
N |
無號短整數 (卅二位, 高位在後的順序) |
V |
無號短整數 (卅二位, 低位在後的順序) |
f |
單精確浮點數 (依電腦的範圍) |
d |
倍精確浮點數 (依電腦的範圍) |
x |
空位 |
X |
倒回一位 |
@ |
填入 NULL 字元到絕對位置 |
unpark函數說明:本函數用來將位的字串的資料解壓縮
文法:unpack(format,args+)
參數 |
描述 |
format |
必需。規定在封裝資料時所使用的格式。 |
args+ |
可選。規定被封裝的一個或多個參數。 |
參數與park相同。
ord函數說明:返回對應字元的acill碼值
文法:ord($character);
執行個體說明:
複製代碼 代碼如下:
<?php
//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";
}
通過上面執行個體,我們可以看到,相同字串,用不同格式儲存,所佔用的位元組數不同。 這裡也可以看到,以不同格式儲存字元可以達儲存節省空間的。而且啟到不可讀加密效果。 突然想到一點,設計資料庫欄位類型問題,如果一個欄位只是:10位長度整型。我們設定為整形:256*256*256*256 就4個位元組,如果設定為10個長度字串。那就佔10個位元組。整個消化空間就是2倍的。 設定正確字元類型對提高資料庫效能有很多協助。呵呵,有點跑題了……
php處理位元組碼通訊執行個體分析
剛剛說的pack作用:節省空間的、加密格式
下面就這2個做一個執行個體說明,介面開發要求:
參數 |
描述 |
使用者名稱 |
20位元組,字元型 |
密碼 |
10位元組,字元型 |
年齡 |
1位元組,無符char型 |
出生年月 |
4位元組,整型(19800101) |
郵箱 |
50位元組,字串 |
各欄位間用:"\0"分割 |
A、PACK封包
複製代碼 代碼如下:
<?php
$code=array(
"username"=>array("A20","張三adfb12"),
"pass"=>array("A10","asdf*#1"),
"age"=>array("C","23"),
"birthday"=>array("I","19900101"),
"email"=>array("A50","zhangsan@163.com"));
$stream=join("\0",parkByArr($code));
echo $stream,strlen($stream);
複製代碼 代碼如下:
file_put_contents("c:/1.txt",$stream);//將流儲存起來便於下面讀取
function parkByArr($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";
}
}
因為用”\0”分割,整個長度是89位元組。通過上面輸出,有一些字串輸出是可以讀取的,其它都已經變成亂碼了。這也是我說可以保密效果原因。
B、Unpack解包
解包需要按照,打包方式讀取,該讀取多長,該用什麼類型讀取,必須與打包規定一樣。
複製代碼 代碼如下:
<?php
$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(parkByArr($stream,$code));
function parkByArr($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;
}