PHP Conference Japan 2024

压缩过滤器

虽然 压缩封装器 提供了一种在本地文件系统上创建与 gzip 和 bz2 兼容的文件的方法,但它们不提供在网络流上进行通用压缩的方法,也不提供从非压缩流开始并转换为压缩流的方法。为此,可以随时将压缩过滤器应用于任何流资源。

注意压缩过滤器不会生成命令行实用程序(如gzip)使用的标头和尾部。它们仅压缩和解压缩压缩数据流的有效负载部分。

zlib.deflate 和 zlib.inflate

zlib.deflate(压缩)和zlib.inflate(解压缩)是 » RFC 1951 中描述的压缩方法的实现。 deflate 过滤器最多接受三个作为关联数组传递的参数。 level 描述要使用的压缩强度(1-9)。较高的数字通常会产生较小的有效负载,但代价是需要额外的处理时间。还存在两个特殊的压缩级别:0(完全不压缩)和 -1(zlib 内部默认值 - 当前为 6)。 window 是压缩循环回环窗口大小的以 2 为底的对数。较高的值(最多 15 - 32768 字节)会产生更好的压缩,但代价是需要更多的内存,而较低的值(低至 9 - 512 字节)会在较小的内存占用空间内产生较差的压缩。默认的 window 大小目前为 15memory 是一个比例,指示应分配多少工作内存。有效值范围从 1(最小分配)到 9(最大分配)。此内存分配仅影响速度,不会影响生成的有效负载的大小。

注意由于压缩级别是最常用的参数,因此可以将其作为简单整数(而不是数组元素)提供。

如果启用了 zlib 支持,则可以使用 zlib.* 压缩过滤器。

示例 #1 zlib.deflatezlib.inflate

<?php
$params
= array('level' => 6, 'window' => 15, 'memory' => 9);

$original_text = "This is a test.\nThis is only a test.\nThis is not an important string.\n";
echo
"The original text is " . strlen($original_text) . " characters long.\n";

$fp = fopen('test.deflated', 'w');
stream_filter_append($fp, 'zlib.deflate', STREAM_FILTER_WRITE, $params);
fwrite($fp, $original_text);
fclose($fp);

echo
"The compressed file is " . filesize('test.deflated') . " bytes long.\n";
echo
"The original text was:\n";
/* Use readfile and zlib.inflate to decompress on the fly */
readfile('php://filter/zlib.inflate/resource=test.deflated');

/* Generates output:

The original text is 70 characters long.
The compressed file is 56 bytes long.
The original text was:
This is a test.
This is only a test.
This is not an important string.

*/
?>

示例 #2 zlib.deflate 简单

<?php
$original_text
= "This is a test.\nThis is only a test.\nThis is not an important string.\n";
echo
"The original text is " . strlen($original_text) . " characters long.\n";

$fp = fopen('test.deflated', 'w');
/* Here "6" indicates compression level 6 */
stream_filter_append($fp, 'zlib.deflate', STREAM_FILTER_WRITE, 6);
fwrite($fp, $original_text);
fclose($fp);

echo
"The compressed file is " . filesize('test.deflated') . " bytes long.\n";

/* Generates output:

The original text is 70 characters long.
The compressed file is 56 bytes long.

*/
?>

bzip2.compress 和 bzip2.decompress

bzip2.compressbzip2.decompress 的工作方式与上面描述的 zlib 过滤器相同。 bzip2.compress 过滤器接受最多两个参数,作为关联数组的元素给出:blocks 是一个整数,取值范围为 1 到 9,指定为工作区分配的 100kbyte 块数。 work 也是一个整数,取值范围为 0 到 250,指示在回退到更慢但更可靠的方法之前,使用普通压缩方法花费多少精力。调整此参数仅影响压缩速度。此设置不会更改压缩输出的大小或内存使用情况。工作因子为 0 指示 bzip 库使用内部默认值。 bzip2.decompress 过滤器仅接受一个参数,该参数可以作为普通布尔值或作为关联数组的 small 元素传递。 small,当设置为 **true** 值时,指示 bzip 库以牺牲速度为代价,以最小的内存占用量执行解压缩。

如果启用了 bz2 支持,则可以使用 bzip2.* 压缩过滤器。

示例 #3 bzip2.compressbzip2.decompress

<?php
$param
= array('blocks' => 9, 'work' => 0);

echo
"The original file is " . filesize('LICENSE') . " bytes long.\n";

$fp = fopen('LICENSE.compressed', 'w');
stream_filter_append($fp, 'bzip2.compress', STREAM_FILTER_WRITE, $param);
fwrite($fp, file_get_contents('LICENSE'));
fclose($fp);

echo
"The compressed file is " . filesize('LICENSE.compressed') . " bytes long.\n";

/* Generates output:

The original file is 3288 bytes long.
The compressed file is 1488 bytes long.

*/
?>
添加注释

用户贡献的笔记 4 条笔记

匿名
9 年前
读取来自 http 的 gzip 编码流
<?php
$opts
= [
"http" => [
"method" => "GET",
"header" => [ "Accept-Encoding: gzip" ],
]
];
$ctx = stream_context_create($opts);
$f = fopen("https://php.net", "r", false, $ctx);
// 检查 stream_get_meta_data($f)["wrapper_data"] 是否包含 "Content-Encoding: gzip"
stream_filter_append($f, "zlib.inflate", STREAM_FILTER_READ, ["window" => 30]);
echo
stream_get_contents($f); // 任何流处理
fclose($f);
匿名
3 年前
要将 zlib.inflate 过滤器与最初使用 gzcompress() 或 zlib.deflate 编写的數據一起使用,请将 window 选项设置为 15,如下所述: https://bugs.php.net/bug.php?id=68556

<?php
$fh
= fopen(file_name, 'rb');
stream_filter_append($fh, 'zlib.inflate', STREAM_FILTER_READ, ['window' => 15]);
$contents = stream_get_contents($fh);
fclose($fh);
bohwaz
6 年前
请注意,此功能目前存在错误。无法使用 ftell()、fseek() 和 fstat() 函数。在使用此函数后写入流不会像预期的那样更改流位置。

请参阅错误: https://bugs.php.net/bug.php?id=49874

此外,zlib 过滤器不适用于 php://temp、php://memory 和 php://input 流,这些流不会输出任何内容。
TingSong
1 年前
解压缩 gzip 流

<?php
$stream
= fopen('https://example.com/some/file.txt.gz', 'rb');
stream_filter_append($stream, 'zlib.inflate', STREAM_FILTER_READ, ['window' => 15+16]);

// 直接读取解压缩的行
$line = fgets($stream);

// 处理行
?>

如 zlib 文档 https://www.zlib.net/manual.html#Advanced

'window' 参数在 8 到 15 之间指定窗口大小从 2⁸ 到 2¹⁵ 字节。可以将其加 16 以使用 gzip 标头和尾部而不是 zlib 包装器进行包装。

并且,window 可以是 -8..-15 用于解包原始 deflate 数据。
To Top