在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
Pack given arguments into a binary string according to The idea for this function was taken from Perl and all formatting codes work the same as in Perl. However, there are some formatting codes that are missing such as Perl's "u" format code. Note that the distinction between signed and unsigned values only affects the function unpack(), where as function pack()gives the same result for signed and unsigned format codes. 参数 描述 The Currently implemented formats are: http://www.php.net/manual/zh/function.pack.php php文档:http://php.net/manual/zh/function.pack.php
<?php The resulting binary string will be 6 bytes long and contain the byte sequence 0x12, 0x34, 0x78, 0x56, 0x41, 0x42.
unpack函数说明:本函数用来将位的字符串的资料解压缩
语法:unpack(format,args+) 参数 描述 format 必需。规定在包装数据时所使用的格式。 args+ 可选。规定被包装的一个或多个参数。 参数与pack相同。 ord函数(ord函数:为序数函数,函数返回值为字符在ASCII码中的序号。)说明:返回对应字符的acill码值 语法:ord($character); //A字符 $str = (pack ( "A*", "中国" )); echo $str, "=", strlen ( $str ), "字节\n"; getAscill ( $str ); echo '<br/>'; //H字符 $str = (pack ( "H*", "fffe" )); echo $str, "=", strlen ( $str ), "字节\n"; getAscill ( $str ); echo '<br/>'; //C字符 $str = (pack ( "C*", "55", "56", "57" )); echo $str, "=", strlen ( $str ), "字节\n"; getAscill ( $str ); echo '<br/>'; //i字符 短整形 32位 4个字节 64位8个字节 $str = (pack ( "i", "100" )); echo $str, "=", strlen ( $str ), "字节\n"; getAscill ( $str ); echo '<br/>'; //s字符 短整形 2个字节 $str = (pack ( "s", "100" )); echo $str, "=", strlen ( $str ), "字节\n"; getAscill ( $str ); echo '<br/>'; //l字符 长整形 4个字节 $str = (pack ( "l", "100" )); echo $str, "=", strlen ( $str ), "字节\n"; getAscill ( $str ); echo '<br/>'; //f字符 单精度浮点 4个字节 $str = (pack ( "f", "100" )); echo $str, "=", strlen ( $str ), "字节\n"; getAscill ( $str ); echo '<br/>'; //d字符 双精度浮点 8个字节 $str = (pack ( "d", "100" )); echo $str, "=", strlen ( $str ), "字节\n"; getAscill ( $str ); echo '<br/>'; function getAscill($str) { $arr = str_split ( $str ); foreach ( $arr as $v ) { echo $v, "=", ord ( $v ), "\n"; } echo "=============\r\n\r\n"; } string '4xVAB' (length=6)
ä¸å›½=6å—节 ä=228 ¸=184 =173 å=229 ›=155 ½=189 =============
通过上面实例,我们可以看到,相同字符串,用不同格式存储,所占用的字节数不同。 这里也可以看到,以不同格式保存字符可以达存储节省空间。而且启到不可读加密效果。 突然想到一点,设计数据库字段类型问题,如果一个字段只是:10位长度整型。我们设置为整形:256*256*256*256 就4个字节,如果设置为10个长度字符串。那就占10个字节。整个消化空间就是2倍的。 设置正确字符类型对提高数据库性能有很多帮助。呵呵,有点跑题了…… php处理字节码通讯实例分析 $code=array( "username"=>array("A20","张三adfb12"), "pass"=>array("A10","asdf*#1"), "age"=>array("C","23"), "birthday"=>array("I","19900101"), "email"=>array("A50","[email protected]")); $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"; } } (在浏览器中用utf8编码查看:张三adfb12 asdf*#1 Ŧ/[email protected] 89; 文件内容: 张三adfb12 asdf*#1 Ŧ/ [email protected] ) 因为用”\0”分割,整个长度是89字节。通过上面输出,有一些字符串输出是可以读取的,其它都已经变成乱码了。这也是我说可以保密效果原因。 <?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(unpackByArr($stream,$code)); function unpackByArr($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; } 参考:http://randomclan.blog.163.com/blog/static/1453009820125454418912/ ------------------------------------------------------- 案例一、pack实现缩减文件数据存储大小
此时test.txt的文件大小是10byte。 注意此时文件大小是10字节,实际占用空间大小是1KB 。
上面存储的整数实际是以字符串形式存储于文件test.txt中。
但如果以整数的二进制字符串存jy储,将会缩减至4byte。
案例二、数据加密
以字符串形式存储一段有意义数据,7-110-abcdefg-117。
字符"-"分割后,第一位表示字符串长度,第二位表示存储位置,第三位表示实际存储的字符串,第四位表示结尾位置。
上述方法缺点:
一、数据存储大小
二、数据以明文方式存储,如果是任何敏感信息,都可能造成不安全访问。
三、文件存储大小,以不规则方式递增。
加密:
存储一段数据,加密格式为:整数2位长度字符串10位长度整数1位长度。
优点:
一、数据大小最优化
二、在不知道"i2a7i1"这样的压缩格式时,即使拿到文件,也无法正确读出二进制文件转化为明文。
三、数据增加时,文件存储大小是等量递增。每次都是以19byte递增。
案例三、key-value型文件存储
存储生成的文件为两个:索引文件,数据文件
文件中数据存储的格式如下图:
<?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, '<'.'?php exit()?'.'>');//定位文件流至起始位置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, '<'.'?php exit()?'.'>');//定位文件流至起始位置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 < $index_count ; $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通信加密
通信双方都定义好加密格式:
例如:
$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解码方式,得到正确的数据. 转自:http://blog.csdn.net/a600423444/article/details/8653827 详细: http://my.oschina.net/goal/blog/195749 http://my.oschina.net/goal/blog/202378?p={{currentPage-1}} 一篇文章: socket字节流解析(网络抓包解析) 研究了一下PHP和C++socket通讯,用C++作为服务器端,php作为客户端进行. socket通讯是基于协议的,因此,只要双方协议一致就行. 关于协议的选择:我看过网上大部分协议都是在应用层的协议,选用这样的协议很方便,基本上就是字符串传过来,传过去 本次研究的协议算是当今国际化的一个标准做法.length+flag+body(长度+类型+内容)的方式,
php实现方式,也很容易,通过pack打包成二进制进行通讯.下面贴一下代码 本地测试主要应用为:发送账号和密码给服务器端 <?php class Byte{ //长度 private $length=0; private $byte=''; //操作码 private $code; public function setBytePrev($content){ $this->byte=$content.$this->byte; } public function getByte(){ return $this->byte; } public function getLength(){ return $this->length; } public function writeChar($string){ $this->length+=strlen($string); $str=array_map('ord',str_split($string)); foreach($str as $vo){ $this->byte.=pack('c',$vo); } $this->byte.=pack('c','0'); $this->length++; } public function writeInt($str){ $this->length+=4; $this->byte.=pack('L',$str); } public function writeShortInt($interge){ $this->length+=2; $this->byte.=pack('v',$interge); } } class GameSocket{ private $socket; private $port=9991; private $host='192.168.211.231'; private $byte; private $code; const CODE_LENGTH=2; const FLAG_LENGTH=4; public function __set($name,$value){ $this->$name=$value; } public function __construct($host='192.168.211.231',$port=9991){ $this->host=$host; $this->port=$port; $this->socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP); if(!$this->socket){ exit('创建socket失败'); } $result = socket_connect($this->socket,$this->host,$this->port); if(!$result){ exit('连接不上目标主机'.$this->host); } $this->byte=new Byte(); } public function write($data){ if(is_string($data)||is_int($data)||is_float($data)){ $data[]=$data; } if(is_array($data)){ foreach($data as $vo){ $this->byte->writeShortInt(strlen($vo)); $this->byte->writeChar($vo); } } |
2022-08-30
2022-08-15
2022-08-17
2022-11-06
2022-08-17
请发表评论