PHP中的gzcompress、gzdeflate、gzencode函数详解
PHP中存在一组看起来很像的压缩解压函数:
压缩函数:gzcompressgzdeflategzencode
解压函数:gzuncompressgzinflategzdecode
gzdecode是PHP5.4.0之后才加入的,使用的时候要注意兼容性问题。
这几个函数都以gz开头,让人想到gzip压缩,而光看函数名却又看不出它们之间的区别,只能查文档。
gzcompressgzdeflategzencode函数的区别在于它们压缩的数据格式不同:
gzcompress使用的是ZLIB格式;
gzdeflate使用的是纯粹的DEFLATE格式;
gzencode使用的是GZIP格式;
但是有一点是相同的,它们压缩数据时都使用了DEFLATE压缩算法(理论上ZLIB和GZIP格式可以使用其他的压缩算法,但是目前实践中只使用DEFLATE算法),ZLIB和GZIP只不过是在DEFLATE的基础之上加了一些头部和尾部而已。
顺便提一下,HTTP协议中的Content-Encoding:deflate使用的是ZLIB格式而不是纯DEFLATE格式。
从PHP5.4.0开始,gzcompress和gzdeflate函数加入了第三个参数$encoding,可以是三个常量:
ZLIB_ENCODING_RAW对应于纯DEFLATE格式;
ZLIB_ENCODING_GZIP对应于GZIP格式;
ZLIB_ENCODING_DEFLATE对应于ZLIB格式(注意不是纯DEFLATE格式);
虽然文档没有提及,但是这三个常量也可以用在gzencode函数的第三个参数$encoding_mode中。
其实从PHP5.4.0开始,这三个函数是一样的,只不过第三个参数的默认值不同;如果调用时传入第三个参数,那么这三个函数返回的数据相同。可以写一个简单的脚本测试:
<?php $url='http://jb51.net'; $s1=gzdeflate($url,1); $s2=gzencode($url,1,ZLIB_ENCODING_RAW); if(strcmp($s1,$s2)==0)echo'thesame'; ?>
运行可以看到$s1和$s2是相同的,为什么会这样呢?可以从PHP源码中找到答案,打开php-5.5.4\ext\zip\zlib.c,可以找到这样的代码:
#definePHP_ZLIB_ENCODE_FUNC(name,default_encoding)\ staticPHP_FUNCTION(name)\ {\ char*in_buf,*out_buf;\ intin_len;\ size_tout_len;\ longlevel=-1;\ longencoding=default_encoding;\ if(default_encoding){\ if(SUCCESS!=zend_parse_parameters(ZEND_NUM_ARGS()TSRMLS_CC,"s|ll",&in_buf,&in_len,&level,&encoding)){\ return;\ }\ }else{\ if(SUCCESS!=zend_parse_parameters(ZEND_NUM_ARGS()TSRMLS_CC,"sl|l",&in_buf,&in_len,&encoding,&level)){\ return;\ }\ }\ if(level<-1||level>9){\ php_error_docref(NULLTSRMLS_CC,E_WARNING,"compressionlevel(%ld)mustbewithin-1..9",level);\ RETURN_FALSE;\ }\ switch(encoding){\ casePHP_ZLIB_ENCODING_RAW:\ casePHP_ZLIB_ENCODING_GZIP:\ casePHP_ZLIB_ENCODING_DEFLATE:\ break;\ default:\ php_error_docref(NULLTSRMLS_CC,E_WARNING,"encodingmodemustbeeitherZLIB_ENCODING_RAW,ZLIB_ENCODING_GZIPorZLIB_ENCODING_DEFLATE");\ RETURN_FALSE;\ }\ if(SUCCESS!=php_zlib_encode(in_buf,in_len,&out_buf,&out_len,encoding,levelTSRMLS_CC)){\ RETURN_FALSE;\ }\ RETURN_STRINGL(out_buf,out_len,0);\ }
/*NOTE:Thenamingoftheseuserlandfunctionswasquiteunlucky*/ /*{{{protobinarygzdeflate(binarydata[,intlevel=-1[,intencoding=ZLIB_ENCODING_RAW]) Encodedatawiththerawdeflateencoding*/ PHP_ZLIB_ENCODE_FUNC(gzdeflate,PHP_ZLIB_ENCODING_RAW); /*}}}*/
/*{{{protobinarygzencode(binarydata[,intlevel=-1[,intencoding=ZLIB_ENCODING_GZIP]) Encodedatawiththegzipencoding*/ PHP_ZLIB_ENCODE_FUNC(gzencode,PHP_ZLIB_ENCODING_GZIP); /*}}}*/
/*{{{protobinarygzcompress(binarydata[,intlevel=-1[,intencoding=ZLIB_ENCODING_DEFLATE]) Encodedatawiththezlibencoding*/ PHP_ZLIB_ENCODE_FUNC(gzcompress,PHP_ZLIB_ENCODING_DEFLATE); /*}}}*/