ホームページ  >  記事  >  バックエンド開発  >  PHP データの圧縮、暗号化、復号化 (パック、アンパック)_PHP チュートリアル

PHP データの圧縮、暗号化、復号化 (パック、アンパック)_PHP チュートリアル

WBOY
WBOYオリジナル
2016-07-14 10:09:281197ブラウズ

ネットワーク通信やファイル ストレージではデータが交換されることが多く、ネットワーク通信トラフィック、ファイル ストレージ サイズ、および暗号化された通信ルールを削減するために、データのセキュリティを確保するためにデータの双方向の暗号化と復号化を実行する必要があることがよくあります。

この関数を PHP で実装するために必要な主な関数は、主に Pack 関数と unpack 関数です
パック
データをビット文字列に圧縮します。
構文: 文字列パック(文字列形式、混​​合[引数]...);
戻り値: 文字列
この関数は、データを圧縮して文字列にパッケージ化するために使用されます。
a - NUL- 埋め込み文字列 [埋め込み文字列] 文字列の空白を NULL 文字で埋め込みます
A - SPACE- パッド付き文字列 [パッド付き文字列]
h – 16 進文字列、下位ニブルが最初 (下位ニブルが最初)
H - 16 進文字列、高「ニブルファースト」(高ニブルファースト)
c – 記号付きの文字
C – 記号のない文字
s – 符号付きの短いパターン (通常は 16 ビット、マシンバイトオーダー)
S – 符号なしの短いパターン (通常は 16 ビット、マシンバイトソート)
n - 符号なしの短いパターン (通常は 16 ビット、ビッグエンディアンバイトソート)
v - 符号なしの短いパターン (通常は 16 ビット、リトルエンディアンのバイトソート)
i – 符号付き整数 (サイズとバイトオーダーによって決定される)
I – 符号なし整数 (サイズとバイトオーダーによって決定される)
l – 署名付きロングパターン [long] (通常は 32 ビット、マシンバイトオーダー)
L – 符号なしロングパターン [long] (通常は 32 ビット、マシンバイトオーダー)
N – 符号なしロングモード [long] (通常は 32 ビット、ビッグエディアンバイトオーダー)
V – 符号なしロングモード [long] (通常は 32 ビット、リトルエディアンバイトオーダー)
f – 浮動小数点 (サイズとバイトオーダーによって決定されます)
d – Double (サイズとバイトオーダーによって決定)
x – null バイト [NUL バイト]
X- 1バイトバックアップ [1バイトバックアップ]
開梱
ビット文字列データを解凍します。
構文: 文字列パック(文字列形式、混​​合[引数]...);
戻り値: 配列
この関数はビット列データを解凍するために使用されます。この関数の機能と使用法は、同じ名前の Perl 関数とまったく同じです。
ケース1、パックによりファイルデータのストレージサイズが削減されます
[php]
//ストレージ整数 1234567890
file_put_contents("test.txt", 1234567890);
//ストレージ整数 1234567890
file_put_contents("test.txt", 1234567890); このとき、test.txtのファイルサイズは10byteです。なお、この時点のファイルサイズは10バイトで、実際に占有される容量は1KBです。
上記で保存された整数は、実際には文字列の形式でファイル test.txt に保存されます。
ただし、整数バイナリ文字列として保存すると、4 バイトに減ります。
[php]
print_r(unpack("i", file_get_contents("test.txt")));
print_r(unpack("i", file_get_contents("test.txt")));
ケース 2、データ暗号化
意味のあるデータを文字列形式 7-110-abcdefg-117 で保存します。
文字「-」を分割した後の最初の桁は文字列の長さを示し、2桁目は格納場所を示し、3桁目は実際に格納された文字列を示し、4桁目は終了位置を示します。
[php]
file_put_contents("test.txt", "7-110-abcdefg-117");
file_put_contents("test.txt", "7-110-abcdefg-117");
上記の方法の欠点:
1. データストレージサイズ
2. データは平文で保存されているため、安全でないアクセスが発生する可能性があります。
3. ファイルストレージのサイズが不規則に増加します。
暗号化:
[php]
file_put_contents("test.txt", Pack("i2a7i1", 7, 110, "abcdefg", 117));
file_put_contents("test.txt", Pack("i2a7i1", 7, 110, "abcdefg", 117));
データを保存する場合、暗号化形式は次のとおりです: 整数 2 桁の長さの文字列 10 桁の長さの整数 1 桁の長さ。
利点:
1. データサイズの最適化
2.「i2a7i1」などの圧縮形式が分からない場合、ファイルを取得してもバイナリファイルを正しく読み込んで平文に変換することができません。
3. データが増加すると、ファイルのストレージサイズも同じ量だけ増加します。毎回 19 バイトずつ増加します。
ケース 3、Key-Value ファイル ストレージ www.2cto.com
生成されたファイルはインデックスファイルとデータファイルの2つに保存されます
ファイル内のデータ保存形式は以下の通りです:
コードの実装:
[php]
エラー報告(E_ALL);
class fileCacheException extends Exception{
}
//Key-Value型ファイルストレージ
クラスファイルキャッシュ{
プライベート $_file_header_size = 14;
プライベート $_file_index_name
プライベート $_file_data_name
private $_file_index;// インデックス ファイル ハンドル
private $_file_data;//データ ファイル ハンドル
private $_node_struct //インデックスノード構造
private $_inx_node_size = 36 //インデックスノードのサイズ
;
パブリック関数 __construct($file_index="filecache_index.dat", $file_data="filecache_data.dat"){
$this->_node_struct = array(
'next'=>array(1, 'V'),
'prev'=>array(1, 'V'),
'data_offset'=>array(1,'V'),//データストレージの開始位置
'data_size'=>array(1,'V'),//データ長
'ref_count'=>array(1,'V'),//ここに引用、PHP の参照カウント破棄モードを模倣します
'key'=>array(16,'H*'),//ストレージキー
);
$this->_file_index_name = $file_index;
$this->_file_data_name = $file_data;
if(!file_exists($this->_file_index_name)){
$this->_create_index();
}その他{
$this->_file_index = fopen($this->_file_index_name, "rb+");
}
if(!file_exists($this->_file_data_name)){
$this->_create_data();
}その他{
$this->_file_data = fopen($this->_file_data_name, "rb+");//バイナリ ストレージには b の使用が必要です
}
}
//インデックスファイルを作成する
プライベート関数_create_index(){
$this->_file_index = fopen($this->_file_index_name, "wb+");//バイナリ ストレージには b の使用が必要です
throw new fileCacheException("インデックス ファイルを開けませんでした:".$this->_file_index_name);
$this->_index_puts(0, '<'.'?php exit()?'.'>');//ファイルフローを開始位置 0 に配置し、ダウンロードを防ぐために php マークを配置します
$this->_index_puts($this->_file_header_size, Pack("V1", 0));
}
//ストレージファイルを作成する
プライベート関数_create_data(){
$this->_file_data = fopen($this->_file_data_name, "wb+");//バイナリ ストレージには b の使用が必要です
throw new fileCacheException("インデックス ファイルを開けませんでした:".$this->_file_data_name
);
$this->_data_puts(0, '<'.'?php exit()?'.'>');//ファイルフローを開始位置 0 に配置し、ダウンロードを防ぐために php マークを配置します
}
プライベート関数 _index_puts($offset, $data, $length=false){
fseek($this->_file_index, $offset);
if($length)
fputs($this->_file_index, $data, $length);
その他
fputs($this->_file_index, $data);
}
プライベート関数 _data_puts($offset, $data, $length=false){
fseek($this->_file_data, $offset);
if($length)
fputs($this->_file_data, $data, $length);
その他
fputs($this->_file_data, $data);
}
/**
* ファイルロック
* @param $is_block 排他ロックかブロックロックかどうか
*/
プライベート関数 _lock($file_res, $is_block=true){
flock($file_res, $is_block ? LOCK_EX : LOCK_EX|LOCK_NB);
}
プライベート関数_unlock($file_res){
フロック($file_res, LOCK_UN);
}
パブリック関数 add($key, $value){
$key = md5($key);
$value = シリアル化($value);
$this->_lock($this->_file_index, true)
$this->_lock($this->_file_data, true);
fseek($this->_file_index, $this->_file_header_size);
list(, $index_count) = unpack('V1', fread($this->_file_index, 4));
$data_size = filesize($this->_file_data_name);
fseek($ this-&gt; _file_data、$ data_size);
$value_size = strlen($value);
$this->_data_puts(filesize($this->_file_data_name), $value);
$node_data =
Pack("V1V1V1V1V1H32", ($index_count==0) ? 0 : $index_count*$this->_inx_node_size, 0, filesize($this->_file_data_name), strlen($value), 0, $key) ;
$index_count++;
$this->_index_puts($this->_file_header_size, $index_count, 4);  
$this->_index_puts($this->get_new_node_pos($index_count), $node_data);  
$this->_unlock($this->_file_data);  
$this->_unlock($this->_file_index);  
}
パブリック関数 get_new_node_pos($index_count){
return $this->_file_header_size + 4 + $this->_inx_node_size * ($index_count-1);  
}
パブリック関数 get_node($key){
$key = md5($key);  
fseek($this->_file_index, $this->_file_header_size);  
$index_count = fread($this->_file_index, 4);  
if($index_count>0) {
for ($i=0; $i
fseek($this->_file_index, $this->_file_header_size + 4 + $this->_inx_node_size * $i);  
$data = fread($this->_file_index, $this->_inx_node_size);  
$node = unpack("V1next/V1prev/V1data_offset/V1data_size/V1ref_count/H32key", $data);  
if($key == $node['key']){
$node を返します。  
}
}
}その他{
null を返します。  
}
}
パブリック関数 get_data($offset, $length){
fseek($this->_file_data, $offset);  
return unserialize(fread($this->_file_data, $length));  
}
}
//使用方法
$cache = 新しい fileCache();  
$cache->add('abcdefg' , 'testabc');  
$data = $cache->get_node('abcdefg');  
print_r($data);  
echo $cache->get_data($data['data_offset'], $data['data_size']);  
エラー報告(E_ALL);
class fileCacheException extends Exception{
}
//Key-Value 型文書存储
クラスファイルキャッシュ{
プライベート $_file_header_size = 14;
プライベート $_file_index_name;
プライベート $_file_data_name;
private $_file_index;//インデックス文件句柄
private $_file_data;//データ文文句柄
private $_node_struct;//インデックス结点结构体
private $_inx_node_size = 36;//インデックス结点大小
パブリック関数 __construct($file_index="filecache_index.dat", $file_data="filecache_data.dat"){
$this->_node_struct = array(
'next'=>array(1, 'V'),
'prev'=>array(1, 'V'),
'data_offset'=>array(1,'V'),//データ存在储開始位置
'data_size'=>array(1,'V'),//データ长度
C 'ref_count' = & gt; array (1, 'v'), // ここで引用、PHP の参照カウントを模倣します。
'key'=>array(16,'H*'),//ストレージKEY
);
$this->_file_index_name = $file_index;
$this->_file_data_name = $file_data;
if(!file_exists($this->_file_index_name)){
$this->_create_index();
}その他{
$this->_file_index = fopen($this->_file_index_name, "rb+");
}
if(!file_exists($this->_file_data_name)){
$this->_create_data();
}その他{
gt;_file_data = fopen($this->_file_data_name, "rb+");//バイナリ ストレージには b
の使用が必要です
}
}
//インデックスファイルを作成します
プライベート関数_create_index(){
$this->_file_index = fopen($this->_file_index_name, "wb+");//バイナリ ストレージには b
の使用が必要です
if(!$this->_file_index)
throw new fileCacheException("インデックス ファイルを開けませんでした:".$this->_file_index_name);
$this->_index_puts(0, '<'. '?php exit()?'.'>');//ファイルフローを開始位置0に配置し、ダウンロードを防ぐためにphpマークを配置します
$this->_index_puts($this->_file_header_size, Pack("V1", 0));
}
//ストレージファイルを作成します
プライベート関数_create_data(){
$this->_file_data = fopen($this->_file_data_name, "wb+");//バイナリ ストレージには b
の使用が必要です
if(!$this->_file_index)
throw new fileCacheException("インデックス ファイルを開けませんでした:".$this->_file_data_name);
$this->_data_puts(0, '<'.'?php exit()?'.'>');//ファイルフローを開始位置 0 に配置し、ダウンロードを防止するために php マークを配置します
}
プライベート関数 _index_puts($offset, $data, $length=false){
fseek($this->_file_index, $offset);
if($length)
fputs($this->_file_index, $data, $length);
その他
fputs($this->_file_index, $data);
}
プライベート関数 _data_puts($offset, $data, $length=false){
fseek($this->_file_data, $offset);
if($length)
fputs($this->_file_data, $data, $length);
その他
fputs($this->_file_data, $data);
}
/**
* ファイルロック
* @param $is_block 排他ロックかブロックロックかどうか
*/
プライベート関数 _lock($file_res, $is_block=true){
flock($file_res, $is_block ? LOCK_EX : LOCK_EX|LOCK_NB);
}
プライベート関数_unlock($file_res){
群れ($file_res, LOCK_UN);
}
パブリック関数 add($key, $value){
$key = md5($key);
$value = Serialize($value);
$this->_lock($this->_file_index, true);
$this->_lock($this->_file_data, true);
fseek($this->_file_index, $this->_file_header_size);
list(, $index_count) = unpack('V1', fread($this->_file_index, 4));
$data_size = filesize($this->_file_data_name);
fseek($this->_file_data, $data_size);
$value_size = strlen($value);
$this->data_puts(filesize($this->_file_data_name), $value);
$node_data =
Pack("V1V1V1V1V1H32", ($index_count==0) ? 0 : $index_count*$this->_inx_node_size, 0, filesize($this->_file_data_name), strlen($value), 0, $key) ;
$index_count++;
$this->_index_puts($this->_file_header_size, $index_count, 4);
$this->_index_puts($this->get_new_node_pos($index_count), $node_data);
$this->_unlock($this->_file_data);
$this->_unlock($this->_file_index);
}
パブリック関数 get_new_node_pos($index_count){
return $this->_file_header_size + 4 + $this->_inx_node_size * ($index_count-1);
}
パブリック関数 get_node($key){
$key = md5($key);
fseek($this->_file_index, $this->_file_header_size);
$index_count = fread($this->_file_index, 4);
if($index_count>0) {
for ($i=0; $i
fseek($this->_file_index, $this->_file_header_size + 4 + $this->_inx_node_size * $i);
$data = fread($this->_file_index, $this->_inx_node_size);
$node = unpack("V1next/V1prev/V1data_offset/V1data_size/V1ref_count/H32key", $data);
if($key == $node['key']){
return $node;
}
}
}その他{
null を返す;
}
}
パブリック関数 get_data($offset, $length){
fseek($this->_file_data, $offset);
return unserialize(fread($this->_file_data, $length));
}
}
//使用方法
$cache = 新しいファイルキャッシュ();
$cache->add('abcdefg' , 'testabc');
$data = $cache->get_node('abcdefg');
print_r($data);
echo $cache->get_data($data['data_offset'], $data['data_size']);
案例四、ソケット通信加密
通信双方都定义好加密格式:
例:
[php]
LOGIN = array(
'COMMAND'=>array('a30', 'LOGIN'),
'DATA'=>array('a30', 'HELLO')
);  
$LOGOUT = array(
'COMMAND'=>array('a30', 'LOGOUT'),
'DATA'=>array('a30', 'GOOD BYE')
);  
$LOGIN_SUCCESS = array(
'COMMAND'=>array('a30', 'LOGIN_SUCCESS'),
'DATA'=>array('V1', 1)
);  
$LOGOUT_SUCCESS = array(
'COMMAND'=>array('a30', 'LOGIN_SUCCESS'),
'DATA'=>array('V1', time())
);  
$LOGIN = 配列(
'COMMAND'=>array('a30', 'LOGIN'),
'DATA'=>array('a30', 'HELLO')
);
$LOGOUT = 配列(
'COMMAND'=>array('a30', 'LOGOUT'),
'DATA'=>array('a30', 'GOOD BYE')
);
$LOGIN_SUCCESS = 配列(
'COMMAND'=>array('a30', 'LOGIN_SUCCESS'),
'DATA'=>array('V1', 1)
);
$LOGOUT_SUCCESS = 配列(
'COMMAND'=>array('a30', 'LOGIN_SUCCESS'),
'DATA'=>array('V1', time())
); サーバー端末とゲスト端末をCOMMAND形式に基づいて解析し、対応するDATA解読方式で正しいデータを取得します

www.bkjia.com本当http://www.bkjia.com/PHPjc/477666.html技術記事ネットワーク通信、文書保存中は、ネットワーク通信量、文書保存サイズ、および密接な通信を減らすために、データの交換が常に必要であり、データを双方向加解して保護する必要があります...
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。