PHP中的gzcompress、gzdeflate、gzencode函数详解


Posted in PHP onJuly 29, 2014

PHP中存在一组看起来很像的压缩解压函数:

压缩函数:gzcompress gzdeflate gzencode

解压函数:gzuncompress gzinflate gzdecode

gzdecode是PHP 5.4.0之后才加入的,使用的时候要注意兼容性问题。

这几个函数都以gz开头,让人想到gzip压缩,而光看函数名却又看不出它们之间的区别,只能查文档。

gzcompress gzdeflate gzencode函数的区别在于它们压缩的数据格式不同:

gzcompress使用的是ZLIB格式;

gzdeflate使用的是纯粹的DEFLATE格式;

gzencode使用的是GZIP格式;

但是有一点是相同的,它们压缩数据时都使用了DEFLATE压缩算法(理论上ZLIB和GZIP格式可以使用其他的压缩算法,但是目前实践中只使用DEFLATE算法),ZLIB和GZIP只不过是在DEFLATE的基础之上加了一些头部和尾部而已。

顺便提一下,HTTP协议中的Content-Encoding: deflate使用的是ZLIB格式而不是纯DEFLATE格式。

从PHP 5.4.0开始,gzcompress和gzdeflate函数加入了第三个参数$encoding,可以是三个常量:

ZLIB_ENCODING_RAW 对应于纯DEFLATE格式;

ZLIB_ENCODING_GZIP 对应于GZIP格式;

ZLIB_ENCODING_DEFLATE 对应于ZLIB格式(注意不是纯DEFLATE格式);

虽然文档没有提及,但是这三个常量也可以用在gzencode函数的第三个参数$encoding_mode中。

其实从PHP 5.4.0开始,这三个函数是一样的,只不过第三个参数的默认值不同;如果调用时传入第三个参数,那么这三个函数返回的数据相同。可以写一个简单的脚本测试:

<?php

$url = 'http://3water.com';

$s1 = gzdeflate($url, 1);

$s2 = gzencode($url, 1, ZLIB_ENCODING_RAW);

if (strcmp($s1, $s2) == 0) echo 'the same';

?>

运行可以看到$s1和$s2是相同的,为什么会这样呢?可以从PHP源码中找到答案,打开php-5.5.4\ext\zip\zlib.c,可以找到这样的代码:

#define PHP_ZLIB_ENCODE_FUNC(name, default_encoding) \

static PHP_FUNCTION(name) \

{ \

    char *in_buf, *out_buf; \

    int in_len; \

    size_t out_len; \

    long level = -1; \

    long encoding = 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(NULL TSRMLS_CC, E_WARNING, "compression level (%ld) must be within -1..9", level); \

        RETURN_FALSE; \

    } \

    switch (encoding) { \

        case PHP_ZLIB_ENCODING_RAW: \

        case PHP_ZLIB_ENCODING_GZIP: \

        case PHP_ZLIB_ENCODING_DEFLATE: \

            break; \

        default: \

            php_error_docref(NULL TSRMLS_CC, E_WARNING, "encoding mode must be either ZLIB_ENCODING_RAW, ZLIB_ENCODING_GZIP or ZLIB_ENCODING_DEFLATE"); \

            RETURN_FALSE; \

    } \

    if (SUCCESS != php_zlib_encode(in_buf, in_len, &out_buf, &out_len, encoding, level TSRMLS_CC)) { \

        RETURN_FALSE; \

    } \

    RETURN_STRINGL(out_buf, out_len, 0); \

}


/* NOTE: The naming of these userland functions was quite unlucky */

/* {{{ proto binary gzdeflate(binary data[, int level = -1[, int encoding = ZLIB_ENCODING_RAW])

   Encode data with the raw deflate encoding */

PHP_ZLIB_ENCODE_FUNC(gzdeflate, PHP_ZLIB_ENCODING_RAW);

/* }}} */
/* {{{ proto binary gzencode(binary data[, int level = -1[, int encoding = ZLIB_ENCODING_GZIP])

   Encode data with the gzip encoding */

PHP_ZLIB_ENCODE_FUNC(gzencode, PHP_ZLIB_ENCODING_GZIP);

/* }}} */
/* {{{ proto binary gzcompress(binary data[, int level = -1[, int encoding = ZLIB_ENCODING_DEFLATE])

   Encode data with the zlib encoding */

PHP_ZLIB_ENCODE_FUNC(gzcompress, PHP_ZLIB_ENCODING_DEFLATE);

/* }}} */

可以看到,gzdeflate gzencode gzcompress三个函数都是用相同的PHP_ZLIB_ENCODE_FUNC宏定义的(是不是有些泛型的意味?),所以它们当然是相同的。

代码中的注释也承认这几个函数的名字起得不好,至于为什么会用这样的名字就不得而知了。

PHP 相关文章推荐
第三节 定义一个类 [3]
Oct 09 PHP
在普通HTTP上安全地传输密码
Jul 21 PHP
C# Assembly类访问程序集信息
Jun 13 PHP
php中判断文件存在是用file_exists还是is_file的整理
Sep 12 PHP
PHP计算2点经纬度之间的距离代码
Aug 12 PHP
PHP字符串长度计算 - strlen()函数使用介绍
Oct 15 PHP
php5.2以下版本无json_decode函数的解决方法
May 25 PHP
PHP常用正则表达式集锦
Aug 17 PHP
mod_php、FastCGI、PHP-FPM等PHP运行方式对比
Jul 02 PHP
PHP递归获取目录内所有文件的实现方法
Nov 01 PHP
PHP面向对象程序设计方法实例详解
Dec 24 PHP
php设计模式之迭代器模式实例分析【星际争霸游戏案例】
Apr 07 PHP
VB中的RasEnumConnections函数返回632错误解决方法
Jul 29 #PHP
php中的curl_multi系列函数使用例子
Jul 29 #PHP
PHP使用CURL_MULTI实现多线程采集的例子
Jul 29 #PHP
PHP实现把文本中的URL转换为链接的auolink()函数分享
Jul 29 #PHP
PHP自带函数给数字或字符串自动补齐位数
Jul 29 #PHP
Win2003+apache+PHP+SqlServer2008 配置生产环境
Jul 29 #PHP
php实现读取超大文件的方法
Jul 28 #PHP
You might like
ionCube 一款类似zend的PHP加密/解密工具
2010/07/25 PHP
php中处理mysql_fetch_assoc返回来的数组 不用foreach----echo
2011/05/04 PHP
PHP网页游戏学习之Xnova(ogame)源码解读(五)
2014/06/23 PHP
javascript 装载iframe子页面,自适应高度
2009/03/20 Javascript
JavaScript 面向对象的之私有成员和公开成员
2010/05/04 Javascript
jquery 3D球状导航的文章分类
2010/07/06 Javascript
JavaScript判断一个URL链接是否有效的实现方法
2011/10/08 Javascript
js 鼠标移动显示图片的简单实例
2013/12/25 Javascript
nodejs通过phantomjs实现下载网页
2015/05/04 NodeJs
JavaScript实现横向滑出的多级菜单效果
2015/10/09 Javascript
Vue组件模板形式实现对象数组数据循环为树形结构(实例代码)
2017/07/31 Javascript
浅谈在Vue-cli里基于axios封装复用请求
2017/11/06 Javascript
Vue父子模版传值及组件传值的三种方法
2017/11/27 Javascript
js中数组对象去重的两种方法
2019/01/18 Javascript
浅谈Webpack多页应用HMR卡住问题
2019/04/24 Javascript
Mpvue中使用Vant Weapp组件库的方法步骤
2019/05/16 Javascript
小程序如何构建骨架屏
2019/05/29 Javascript
JS如何生成动态列表
2020/09/22 Javascript
react-intl实现React国际化多语言的方法
2020/09/27 Javascript
[01:46]辉夜杯—打造中国DOTA新格局
2015/12/25 DOTA
利用Python绘制MySQL数据图实现数据可视化
2015/03/30 Python
python实现判断数组是否包含指定元素的方法
2015/07/15 Python
python+django快速实现文件上传
2016/10/24 Python
Python实现简单石头剪刀布游戏
2021/01/20 Python
python zip()函数的使用示例
2020/09/23 Python
用60行代码实现Python自动抢微信红包
2021/02/04 Python
吃透移动端 1px的具体用法
2019/12/16 HTML / CSS
linux面试题参考答案(4)
2013/01/28 面试题
小学教育毕业生自荐信
2013/11/18 职场文书
全神贯注教学反思
2014/02/03 职场文书
校长竞聘演讲稿
2014/05/16 职场文书
2014年小学教师工作总结
2014/11/10 职场文书
工作证明格式范文
2015/06/15 职场文书
运动会入场词
2015/07/18 职场文书
python基于OpenCV模板匹配识别图片中的数字
2021/03/31 Python
Golang二维切片初始化的实现
2021/04/08 Golang