0

0

PHP数据压缩、加解密(pack, unpack)_PHP教程

php中文网

php中文网

发布时间:2016-07-14 10:09:28

|

1466人浏览过

|

来源于php中文网

原创

网络通信、文件存储中经常需要交换数据,为了减少网络通信流量、文件存储大小以及加密通信规则,经常需要对数据进行双向加解密以保证数据的安全。

PHP中实现此功能主要需要使用的函数主要是pack及unpack函数
 
 
pack
压缩资料到位字符串之中。
 
语法: string pack(string format, mixed [args]...);
 
返回值: 字符串
本函数用来将资料压缩打包到位的字符串之中。
 
 
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 的同名函数功能用法完全相同。
 
 
案例一、pack实现缩减文件数据存储大小
[php]
//存储整数1234567890   
file_put_contents("test.txt", 1234567890);  
 
//存储整数1234567890
file_put_contents("test.txt", 1234567890);此时test.txt的文件大小是10byte。注意此时文件大小是10字节,实际占用空间大小是1KB。
 
 
 
上面存储的整数实际是以字符串形式存储于文件test.txt中。
但如果以整数的二进制字符串存jy储,将会缩减至4byte。
 
[php] 
print_r(unpack("i", file_get_contents("test.txt")));  
 
print_r(unpack("i", file_get_contents("test.txt")));
 
 
 
 
 
 
案例二、数据加密
以字符串形式存储一段有意义数据,7-110-abcdefg-117。
字符"-"分割后,第一位表示字符串长度,第二位表示存储位置,第三位表示实际存储的字符串,第四位表示结尾位置。
[php] 
file_put_contents("test.txt", "7-110-abcdefg-117");  
 
file_put_contents("test.txt", "7-110-abcdefg-117");
上述方法缺点:
一、数据存储大小
二、数据以明文方式存储,如果是任何敏感信息,都可能造成不安全访问。
三、文件存储大小,以不规则方式递增。
 
 
加密:
[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位长度。
优点:
一、数据大小最优化
二、在不知道"i2a7i1"这样的压缩格式时,即使拿到文件,也无法正确读出二进制文件转化为明文。
三、数据增加时,文件存储大小是等量递增。每次都是以19byte递增。
 
 
案例三、key-value型文件存储www.2cto.com
存储生成的文件为两个:索引文件,数据文件
文件中数据存储的格式如下图:
 
\
代码实现:
[php]
error_reporting(E_ALL);  
  
class fileCacheException extends Exception{  
  
}  
  
//Key-Value型文件存储   
class fileCache{  
     private $_file_header_size = 14;  
     private $_file_index_name;  
     private $_file_data_name;  
     private $_file_index;//索引文件句柄   
     private $_file_data;//数据文件句柄   
     private $_node_struct;//索引结点结构体   
     private $_inx_node_size = 36;//索引结点大小   
  
     public function __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*'),//存储KEY   
          );  
  
          $this->_file_index_name = $file_index;  
          $this->_file_data_name = $file_data;  
  
          if(!file_exists($this->_file_index_name)){  
               $this->_create_index();  
          }else{  
               $this->_file_index = fopen($this->_file_index_name, "rb+");  
          }  
  
          if(!file_exists($this->_file_data_name)){  
               $this->_create_data();  
          }else{  
               $this->_file_data = fopen($this->_file_data_name, "rb+");//二进制存储需要使用b   
          }  
     }  
  
     //创建索引文件   
     private function _create_index(){  
          $this->_file_index = fopen($this->_file_index_name, "wb+");//二进制存储需要使用b   
          if(!$this->_file_index)   
               throw new fileCacheException("Could't open index file:".$this->_file_index_name);  
  
          $this->_index_puts(0, '');//定位文件流至起始位置0, 放置php标记防止下载   
          $this->_index_puts($this->_file_header_size, pack("V1", 0));  
     }  
  
  
     //创建存储文件   
     private function _create_data(){  
          $this->_file_data = fopen($this->_file_data_name, "wb+");//二进制存储需要使用b   
          if(!$this->_file_index)   
               throw new fileCacheException("Could't open index file:".$this->_file_data_name);  
  
          $this->_data_puts(0, '');//定位文件流至起始位置0, 放置php标记防止下载   
     }  
  
     private function _index_puts($offset, $data, $length=false){  
          fseek($this->_file_index, $offset);  
  
          if($length)  
          fputs($this->_file_index, $data, $length);  
          else  
          fputs($this->_file_index, $data);  
     }  
  
     private function _data_puts($offset, $data, $length=false){  
          fseek($this->_file_data, $offset);  
          if($length)  
          fputs($this->_file_data, $data, $length);  
          else  
          fputs($this->_file_data, $data);  
     }  
  
     /** 
     * 文件锁 
     * @param $is_block 是否独占、阻塞锁 
     */  
     private function _lock($file_res, $is_block=true){  
          flock($file_res, $is_block ? LOCK_EX : LOCK_EX|LOCK_NB);  
     }  
  
     private function _unlock($file_res){  
          flock($file_res, LOCK_UN);  
     }  
  
     public function 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);  
     }  
  
     public function get_new_node_pos($index_count){  
          return $this->_file_header_size + 4 + $this->_inx_node_size * ($index_count-1);  
     }  
  
     public function 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;  
                    }  
               }  
          }else{  
               return null;  
          }  
     }  
  
     public function get_data($offset, $length){  
          fseek($this->_file_data, $offset);  
          return unserialize(fread($this->_file_data, $length));  
     }  
}  
  
//使用方法   
$cache = new fileCache();  
$cache->add('abcdefg' , 'testabc');  
$data = $cache->get_node('abcdefg');  
print_r($data);  
echo $cache->get_data($data['data_offset'], $data['data_size']);  
 
error_reporting(E_ALL);
 
class fileCacheException extends Exception{
 
}
 
//Key-Value型文件存储
class fileCache{
     private $_file_header_size = 14;
     private $_file_index_name;
     private $_file_data_name;
     private $_file_index;//索引文件句柄
     private $_file_data;//数据文件句柄
     private $_node_struct;//索引结点结构体
     private $_inx_node_size = 36;//索引结点大小
 
     public function __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*'),//存储KEY
          );
 
          $this->_file_index_name = $file_index;
          $this->_file_data_name = $file_data;
 
          if(!file_exists($this->_file_index_name)){
               $this->_create_index();
          }else{
               $this->_file_index = fopen($this->_file_index_name, "rb+");
          }
 
          if(!file_exists($this->_file_data_name)){
               $this->_create_data();
          }else{
               $this->_file_data = fopen($this->_file_data_name, "rb+");//二进制存储需要使用b
          }
     }
 
     //创建索引文件
     private function _create_index(){
          $this->_file_index = fopen($this->_file_index_name, "wb+");//二进制存储需要使用b
          if(!$this->_file_index) 
               throw new fileCacheException("Could't open index file:".$this->_file_index_name);
 
          $this->_index_puts(0, '');//定位文件流至起始位置0, 放置php标记防止下载
          $this->_index_puts($this->_file_header_size, pack("V1", 0));
     }
 
 
     //创建存储文件
     private function _create_data(){
          $this->_file_data = fopen($this->_file_data_name, "wb+");//二进制存储需要使用b
          if(!$this->_file_index) 
               throw new fileCacheException("Could't open index file:".$this->_file_data_name);
 
          $this->_data_puts(0, '');//定位文件流至起始位置0, 放置php标记防止下载
     }
 
     private function _index_puts($offset, $data, $length=false){
          fseek($this->_file_index, $offset);
 
          if($length)
          fputs($this->_file_index, $data, $length);
          else
          fputs($this->_file_index, $data);
     }
 
     private function _data_puts($offset, $data, $length=false){
          fseek($this->_file_data, $offset);
          if($length)
          fputs($this->_file_data, $data, $length);
          else
          fputs($this->_file_data, $data);
     }
 
     /**
     * 文件锁
     * @param $is_block 是否独占、阻塞锁
     */
     private function _lock($file_res, $is_block=true){
          flock($file_res, $is_block ? LOCK_EX : LOCK_EX|LOCK_NB);
     }
 
     private function _unlock($file_res){
          flock($file_res, LOCK_UN);
     }
 
     public function 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);
     }
 
     public function get_new_node_pos($index_count){
          return $this->_file_header_size + 4 + $this->_inx_node_size * ($index_count-1);
     }
 
     public function 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;
                    }
               }
          }else{
               return null;
          }
     }
 
     public function get_data($offset, $length){
          fseek($this->_file_data, $offset);
          return unserialize(fread($this->_file_data, $length));
     }
}
 
//使用方法
$cache = new fileCache();
$cache->add('abcdefg' , 'testabc');
$data = $cache->get_node('abcdefg');
print_r($data);
echo $cache->get_data($data['data_offset'], $data['data_size']);
 
 
 
 
案例四、socket通信加密
通信双方都定义好加密格式:
例如:
[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 = 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())
);服务器端与客户端根据解析COMMAND格式,找到对应的DATA解码方式,得到正确的数据
 

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/477666.htmlTechArticle网络通信、文件存储中经常需要交换数据,为了减少网络通信流量、文件存储大小以及加密通信规则,经常需要对数据进行双向加解密以保...

相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

相关标签:

php

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

928

2026.02.13

微博网页版主页入口与登录指南_官方网页端快速访问方法
微博网页版主页入口与登录指南_官方网页端快速访问方法

本专题系统整理微博网页版官方入口及网页端登录方式,涵盖首页直达地址、账号登录流程与常见访问问题说明,帮助用户快速找到微博官网主页,实现便捷、安全的网页端登录与内容浏览体验。

307

2026.02.13

Flutter跨平台开发与状态管理实战
Flutter跨平台开发与状态管理实战

本专题围绕Flutter框架展开,系统讲解跨平台UI构建原理与状态管理方案。内容涵盖Widget生命周期、路由管理、Provider与Bloc状态管理模式、网络请求封装及性能优化技巧。通过实战项目演示,帮助开发者构建流畅、可维护的跨平台移动应用。

183

2026.02.13

TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

29

2026.02.13

Redis高可用架构与分布式缓存实战
Redis高可用架构与分布式缓存实战

本专题围绕 Redis 在高并发系统中的应用展开,系统讲解主从复制、哨兵机制、Cluster 集群模式及数据分片原理。内容涵盖缓存穿透与雪崩解决方案、分布式锁实现、热点数据优化及持久化策略。通过真实业务场景演示,帮助开发者构建高可用、可扩展的分布式缓存系统。

103

2026.02.13

c语言 数据类型
c语言 数据类型

本专题整合了c语言数据类型相关内容,阅读专题下面的文章了解更多详细内容。

54

2026.02.12

雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法
雨课堂网页版登录入口与使用指南_官方在线教学平台访问方法

本专题系统整理雨课堂网页版官方入口及在线登录方式,涵盖账号登录流程、官方直连入口及平台访问方法说明,帮助师生用户快速进入雨课堂在线教学平台,实现便捷、高效的课程学习与教学管理体验。

17

2026.02.12

豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法
豆包AI网页版入口与智能创作指南_官方在线写作与图片生成使用方法

本专题汇总豆包AI官方网页版入口及在线使用方式,涵盖智能写作工具、图片生成体验入口和官网登录方法,帮助用户快速直达豆包AI平台,高效完成文本创作与AI生图任务,实现便捷智能创作体验。

764

2026.02.12

PostgreSQL性能优化与索引调优实战
PostgreSQL性能优化与索引调优实战

本专题面向后端开发与数据库工程师,深入讲解 PostgreSQL 查询优化原理与索引机制。内容包括执行计划分析、常见索引类型对比、慢查询优化策略、事务隔离级别以及高并发场景下的性能调优技巧。通过实战案例解析,帮助开发者提升数据库响应速度与系统稳定性。

92

2026.02.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 12.2万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 11.3万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号