PHP數據壓縮、加解密(pack, unpack)

網絡通訊、文件存儲中常常須要交換數據,爲了減小網絡通訊流量、文件存儲大小以及加密通訊規則,常常須要對數據進行雙向加解密以保證數據的安全。
PHP中實現此功能主要須要使用的函數主要是pack及unpack函數

pack

壓縮資料到位字符串之中。 php

語法: string pack(string format, mixed [args]...); node

返回值: 字符串
本函數用來將資料壓縮打包到位的字符串之中。

a - NUL- 字符串填滿[padded string] 將字符串空白以 NULL 字符填滿
A - SPACE- 字符串填滿[padded string]
h – 十六進制字符串,低「四位元」[low nibble first] (低位在前)
H - 十六進制字符串,高「四位元」[high nibble first](高位在前)
c – 帶有符號的字符
C – 不帶有符號的字符
s – 帶有符號的短模式[short](一般是16位,按機器字節順序)
S – 不帶有符號的短模式[short](一般是16位,按機器字節排序)
n -不帶有符號的短模式[short](一般是16位,按大endian字節排序)
v -不帶有符號的短模式[short](一般是16位,按小endian字節排序)
i – 帶有符號的整數(由大小和字節順序決定)
I – 不帶有符號的整數(由大小和字節順序決定)
l– 帶有符號的長模式[long](一般是32位,按機器字節順序)
L – 不帶有符號的長模式[long](一般是32位,按機器字節順序)
N – 不帶有符號的長模式[long](一般是32位,按大edian字節順序)
V– 不帶有符號的長模式[long](一般是32位,按小edian字節順序)
f –浮點(由大小和字節順序決定)
d – 雙精度(由大小和字節順序決定)
x – 空字節[NUL byte]
X- 後面一個字節[Back up one byte](倒回一位)


unpack 數組

解壓縮位字符串資料。 安全

語法: string pack(string format, mixed [args]...); 服務器

返回值: 數組
本函數用來將位的字符串的資料解壓縮。本函數和 Perl 的同名函數功能用法徹底相同。

案例1、pack實現縮減文件數據存儲大小
[php]  view plain copy
  1. <?php  
  2. //存儲整數1234567890  
  3. file_put_contents("test.txt", 1234567890);  
此時test.txt的文件大小是10byte。 注意此時文件大小是10字節,實際佔用空間大小是1KB

上面存儲的整數實際是以字符串形式存儲於文件test.txt中。
但若是以整數的二進制字符串存jy儲,將會縮減至4byte。
[php]  view plain copy
  1. <?php  
  2. print_r(unpack("i"file_get_contents("test.txt")));  



案例2、數據加密
以字符串形式存儲一段有意義數據,7-110-abcdefg-117。
字符"-"分割後,第一位表示字符串長度,第二位表示存儲位置,第三位表示實際存儲的字符串,第四位表示結尾位置。
[php]  view plain copy
  1. <?php  
  2. file_put_contents("test.txt""7-110-abcdefg-117");  
上述方法缺點:
1、數據存儲大小
2、數據以明文方式存儲,若是是任何敏感信息,均可能形成不安全訪問。
3、文件存儲大小,以不規則方式遞增。

加密:
[php]  view plain copy
  1. <?php  
  2. file_put_contents("test.txt", pack("i2a7i1", 7, 110, "abcdefg", 117));  
存儲一段數據,加密格式爲:整數2位長度字符串10位長度整數1位長度。
優勢:
1、數據大小最優化
2、在不知道"i2a7i1"這樣的壓縮格式時,即便拿到文件,也沒法正確讀出二進制文件轉化爲明文。
3、數據增長時,文件存儲大小是等量遞增。每次都是以19byte遞增。

案例3、key-value型文件存儲
存儲生成的文件爲兩個:索引文件,數據文件
文件中數據存儲的格式以下圖:

代碼實現:
[php]  view plain copy
  1. <?php  
  2. error_reporting(E_ALL);  
  3.   
  4. class fileCacheException extends Exception{  
  5.   
  6. }  
  7.   
  8. //Key-Value型文件存儲  
  9. class fileCache{  
  10.      private $_file_header_size = 14;  
  11.      private $_file_index_name;  
  12.      private $_file_data_name;  
  13.      private $_file_index;//索引文件句柄  
  14.      private $_file_data;//數據文件句柄  
  15.      private $_node_struct;//索引結點結構體  
  16.      private $_inx_node_size = 36;//索引結點大小  
  17.   
  18.      public function __construct($file_index="filecache_index.dat"$file_data="filecache_data.dat"){  
  19.           $this->_node_struct = array(  
  20.                'next'=>array(1, 'V'),  
  21.                'prev'=>array(1, 'V'),  
  22.               'data_offset'=>array(1,'V'),//數據存儲起始位置  
  23.               'data_size'=>array(1,'V'),//數據長度  
  24.               'ref_count'=>array(1,'V'),//引用此處,模仿PHP的引用計數銷燬模式  
  25.               'key'=>array(16,'H*'),//存儲KEY  
  26.           );  
  27.   
  28.           $this->_file_index_name = $file_index;  
  29.           $this->_file_data_name = $file_data;  
  30.   
  31.           if(!file_exists($this->_file_index_name)){  
  32.                $this->_create_index();  
  33.           }else{  
  34.                $this->_file_index = fopen($this->_file_index_name, "rb+");  
  35.           }  
  36.   
  37.           if(!file_exists($this->_file_data_name)){  
  38.                $this->_create_data();  
  39.           }else{  
  40.                $this->_file_data = fopen($this->_file_data_name, "rb+");//二進制存儲須要使用b  
  41.           }  
  42.      }  
  43.   
  44.      //建立索引文件  
  45.      private function _create_index(){  
  46.           $this->_file_index = fopen($this->_file_index_name, "wb+");//二進制存儲須要使用b  
  47.           if(!$this->_file_index)   
  48.                throw new fileCacheException("Could't open index file:".$this->_file_index_name);  
  49.   
  50.           $this->_index_puts(0, '<'.'?php exit()?'.'>');//定位文件流至起始位置0, 放置php標記防止下載  
  51.           $this->_index_puts($this->_file_header_size, pack("V1", 0));  
  52.      }  
  53.   
  54.   
  55.      //建立存儲文件  
  56.      private function _create_data(){  
  57.           $this->_file_data = fopen($this->_file_data_name, "wb+");//二進制存儲須要使用b  
  58.           if(!$this->_file_index)   
  59.                throw new fileCacheException("Could't open index file:".$this->_file_data_name);  
  60.   
  61.           $this->_data_puts(0, '<'.'?php exit()?'.'>');//定位文件流至起始位置0, 放置php標記防止下載  
  62.      }  
  63.   
  64.      private function _index_puts($offset$data$length=false){  
  65.           fseek($this->_file_index, $offset);  
  66.   
  67.           if($length)  
  68.           fputs($this->_file_index, $data$length);  
  69.           else  
  70.           fputs($this->_file_index, $data);  
  71.      }  
  72.   
  73.      private function _data_puts($offset$data$length=false){  
  74.           fseek($this->_file_data, $offset);  
  75.           if($length)  
  76.           fputs($this->_file_data, $data$length);  
  77.           else  
  78.           fputs($this->_file_data, $data);  
  79.      }  
  80.   
  81.      /** 
  82.      * 文件鎖 
  83.      * @param $is_block 是否獨佔、阻塞鎖 
  84.      */  
  85.      private function _lock($file_res$is_block=true){  
  86.           flock($file_res$is_block ? LOCK_EX : LOCK_EX|LOCK_NB);  
  87.      }  
  88.   
  89.      private function _unlock($file_res){  
  90.           flock($file_res, LOCK_UN);  
  91.      }  
  92.   
  93.      public function add($key$value){  
  94.           $key = md5($key);  
  95.           $value = serialize($value);  
  96.           $this->_lock($this->_file_index, true);  
  97.           $this->_lock($this->_file_data, true);  
  98.   
  99.           fseek($this->_file_index, $this->_file_header_size);  
  100.   
  101.           list(, $index_count) = unpack('V1'fread($this->_file_index, 4));  
  102.   
  103.           $data_size = filesize($this->_file_data_name);  
  104.   
  105.           fseek($this->_file_data, $data_size);  
  106.   
  107.           $value_size = strlen($value);  
  108.   
  109.           $this->_data_puts(filesize($this->_file_data_name), $value);  
  110.   
  111.           $node_data =   
  112.           pack("V1V1V1V1V1H32", ($index_count==0) ? 0 : $index_count*$this->_inx_node_size, 0, filesize($this->_file_data_name), strlen($value), 0, $key);  
  113.   
  114.           $index_count++;  
  115.   
  116.           $this->_index_puts($this->_file_header_size, $index_count, 4);  
  117.   
  118.           $this->_index_puts($this->get_new_node_pos($index_count), $node_data);  
  119.   
  120.           $this->_unlock($this->_file_data);  
  121.           $this->_unlock($this->_file_index);  
  122.      }  
  123.   
  124.      public function get_new_node_pos($index_count){  
  125.           return $this->_file_header_size + 4 + $this->_inx_node_size * ($index_count-1);  
  126.      }  
  127.   
  128.      public function get_node($key){  
  129.           $key = md5($key);  
  130.           fseek($this->_file_index, $this->_file_header_size);  
  131.           $index_count = fread($this->_file_index, 4);  
  132.   
  133.           if($index_count>0) {  
  134.                for ($i=0; $i < $index_count ; $i++) {   
  135.                     fseek($this->_file_index, $this->_file_header_size + 4 + $this->_inx_node_size * $i);  
  136.                     $data = fread($this->_file_index, $this->_inx_node_size);  
  137.                     $node = unpack("V1next/V1prev/V1data_offset/V1data_size/V1ref_count/H32key"$data);  
  138.   
  139.                     if($key == $node['key']){  
  140.                          return $node;  
  141.                     }  
  142.                }  
  143.           }else{  
  144.                return null;  
  145.           }  
  146.      }  
  147.   
  148.      public function get_data($offset$length){  
  149.           fseek($this->_file_data, $offset);  
  150.           return unserialize(fread($this->_file_data, $length));  
  151.      }  
  152. }  
  153.   
  154. //使用方法  
  155. $cache = new fileCache();  
  156. $cache->add('abcdefg' , 'testabc');  
  157. $data = $cache->get_node('abcdefg');  
  158. print_r($data);  
  159. echo $cache->get_data($data['data_offset'], $data['data_size']);  



案例4、socket通訊加密
通訊雙方都定義好加密格式:
例如:
[php]  view plain copy
  1. $LOGIN = array(  
  2.      'COMMAND'=>array('a30''LOGIN'),  
  3.      'DATA'=>array('a30''HELLO')  
  4. );  
  5.   
  6. $LOGOUT = array(  
  7.      'COMMAND'=>array('a30''LOGOUT'),  
  8.      'DATA'=>array('a30''GOOD BYE')  
  9. );  
  10.   
  11. $LOGIN_SUCCESS = array(  
  12.      'COMMAND'=>array('a30''LOGIN_SUCCESS'),  
  13.      'DATA'=>array('V1', 1)  
  14. );  
  15.   
  16. $LOGOUT_SUCCESS = array(  
  17.      'COMMAND'=>array('a30''LOGIN_SUCCESS'),  
  18.      'DATA'=>array('V1', time())  
  19. );  
服務器端與客戶端根據解析COMMAND格式,找到對應的DATA解碼方式,獲得正確的數據
相關文章
相關標籤/搜索