这篇指南就是关于这样一种文件格式的,可能应用程序开发者几乎每天都会遇到这种文件:ZIP格式。一般这种格式是用来通过电子邮件和远程连接传 输文件的,能够将多个文件压缩到一个档案文件中,因此,减少了文件的硬盘占据空间,并且能够更容易地移动它们。PHP通过它的ZZipLib插件和 PEAR的Archive_Zip类都可以读取和创建这些ZIP文件。
我将假设您已经有了正常运行的Apache,安装了PHP,并且PEAR Archive_Zip class类已经正确安装了。
注意:您可以直接从网上安装PEAR Archive_Zip程序包,要么下载它,还可以利用提供的指示。
创建ZIP档案文件
让我们从一个简单的例子开始:动态地创建一个包括几个其他文件的ZIP档案文件。以列表A中的脚本开始。
列表A
- <?php
- include ('Archive/Zip.php'); // imports
- $obj = new Archive_Zip('test.zip'); // name of zip file
- $files = array('mystuff/ad.gif',
- 'mystuff/alcon.doc',
- 'mystuff/alcon.xls'); // files to store
- if ($obj->create($files)) {
- echo 'Created successfully!';
- } else {
- echo 'Error in file creation';
- }
- ?>
这个脚本非常简单,但是值得仔细看一下:
首先,第一步是创建一个Archive_Zip类的实例,然后用将要创建的档案文件的路径和名称将其初始化。在这个例子中,档案文件被命名为test.zip,位于当前目录下。
接着,初始化一个数组,列出将要被压缩的文件,和它们在硬盘中的位置一起保存其中;这些位置用绝对或相对术语列入,但是,一个关键的要考虑的事项是脚本对那些文件或磁盘的位置要有读取的权限。
最后,用create()方法通过压缩和合并指定的文件来实际构建档案文件。这个方法接受文件列表作为输入,然后返回一个布尔逻辑值指示档案文 件是否被成功创建。注意脚本在文件被创建的目录下必须有写入特权是非常重要的,否则create()方法将失败;这是一个普遍的,也让大部分新手失败的错 误。
现在,在修改了源文件列表和目标文件位置来反映您的本地系统配置之后,试着运行上面的脚本。如果一切顺利的话,Archive_Zip应该可以找到您所列出的、并压缩到名为test.zip的ZIP档案文件中的文件。
查看ZIP档案文件的内容
那么怎么样看到现有的ZIP档案文件中的内容呢?Archive_Zip lets让您通过它的listContent()方法也可以做到这一点。下面是一个例子(列表B):
列表B
- <?php
- include ('Archive/Zip.php'); // imports
- $obj = new Archive_Zip('test.zip'); // name of zip file
- $files = $obj->listContent(); // array of file information
- foreach ($files as $f) {
- foreach ($f as $k => $v) {
- echo "$k: $v\n";
- }
- echo "\n";
- }
- ?>
listContent()的输出是一个由数组组成的结构数组,每一个数组元素代表档案文件中的一个单独文件。通常,每一个元素中保存有相关的 信息,例如对应文件的名字、它的索引位置、状态、大小(压缩后和压缩前的)和最近一次修改的时间。用一个循环可以很容易地将这些信息提取出来,还可以像列 表B那样,重定它的格式,使其更好传输。下面是输出的一个示例(列表C):
列表C
- filename: mystuff/alcon.xls
- stored_filename: mystuff/alcon.xls
- size: 113664
- compressed_size: 35902
- mtime: 1141996836
- comment:
- folder:
- index: 0
- status: ok
向现有的ZIP档案文件中添加新文件
Archive_Zip类的一个有意思的特性就是它可以通过add()方法向现有的档案文件中添加新的文件。为了说明这一点,让我们回到test.zip,尝试对它添加新文件(列表D):
列表D
- <?php
- include ('Archive/Zip.php'); // imports
- if (file_exists('test.zip')) {
- $obj = new Archive_Zip('test.zip'); // name of zip file
- } else {
- die('File does not exist');
- }
- $files = array('otherstuff/montecarlo.png'); // additional files to store
- if ($obj->add($files)) {
- echo 'Added successfully!';
- } else {
- echo 'Error in file addition';
- }
- ?>
正如您所看到的那样,向一个现有的档案文件中添加新文件的程序和创建一个新的档案文件十分相似:初始化一个新的Archive_Zip对象指向 问号代表的档案文件,创建一个数组代表将要添加的文件的列表,然后将这个数组输入add()方法。和create()方法一样,add()返回一个布尔逻 辑信号来指示添加是否成功。和前面一样,一个主要的问题就是别忘了要有足够的权限:记得确保脚本有适当的权限来读取源文件,并将新压缩的档案文件写回到硬 盘中。
从现有的ZIP档案文件中删除文件
和添加文件一样,您也可以删除文件。Archive_Zip类具有delete()方法,让您能够从现有的档案文件中移除文件。列表E说明了这一点。.
列表E
- <?php
- include ('Archive/Zip.php'); // imports
- if (file_exists('test.zip')) {
- $obj = new Archive_Zip('test.zip'); // name of zip file
- } else {
- die('File does not exist');
- }
- $files = array('mystuff/ad.gif', 'otherstuff/montecarlo.png'); // files to delete
- if ($obj->delete(array('by_name' => $files))) {
- echo 'Deleted successfully!';
- } else {
- echo 'Error in file deletion';
- }
- ?>
在这里,创建了一个待删除文件的数组,然后将其输入delete()方法。注意delete()调用中的特殊参数“by_name”:这告诉Archive_Zip只删除那些与文件名精确匹配的文件。如果删除成功,delete()方法返回真。
除了这种形式的有选择的删除之外,delete()方法也支持对与特定类型或正则表达式相匹配文件的大规模的摧毁。利用“by_ereg”或 “by_preg”参数,Perl和PHP的正则表达式都支持。列表F是一个例子,用来说明怎样用这种方法,通过利用Perl的正则表达式来删除一个档案 文件中所有的*.doc文件。
列表F
- <?php
- include ('Archive/Zip.php'); // imports
- if (file_exists('test.zip')) {
- $obj = new Archive_Zip('test.zip'); // name of zip file
- } else {
- die('File does not exist');
- }
- if ($obj->delete(array('by_preg' => "/.*doc$/"))) { // all DOC files
- echo 'Deleted successfully!';
- } else {
- echo 'Error in file deletion';
- }
- ?>
如以上的例子所示,PEAR的Archive_Zip类用途很多,只需要几行代码,就使您能够执行一些相当复杂的与ZIP文件的交互。但愿上面的示例脚本能够激发起您的灵感,告诉您如何在您的日常开发活动中使用这个类,并让您对用它进行试验产生兴趣。祝您编程开心