php 遍历文件夹并压成zip_PHP扩展类ZipArchive实现压缩解压Zip文件和文件打包下载...

PHP ZipArchive 是PHP自带的扩展类,可以轻松实现ZIP文件的压缩和解压,使用前首先要确保PHP ZIP 扩展已经开启,具体开启方法就不说了,不同的平台开启PHP扩增的方法网上都有,如有疑问欢迎交流。这里整理一下常用的示例供参考。

一、解压缩zip文件

$zip = new ZipArchive;//新建一个ZipArchive的对象

/*

通过ZipArchive的对象处理zip文件

$zip->open这个方法的参数表示处理的zip文件名。

如果对zip文件对象操作成功,$zip->open这个方法会返回TRUE

*/

if ($zip->open('test.zip') === TRUE)

{

$zip->extractTo('images');//假设解压缩到在当前路径下images文件夹的子文件夹php

$zip->close();//关闭处理的zip文件

}

二、将文件压缩成zip文件

$zip = new ZipArchive;

/*

$zip->open这个方法第一个参数表示处理的zip文件名。

第二个参数表示处理模式,ZipArchive::OVERWRITE表示如果zip文件存在,就覆盖掉原来的zip文件。

如果参数使用ZIPARCHIVE::CREATE,系统就会往原来的zip文件里添加内容。

如果不是为了多次添加内容到zip文件,建议使用ZipArchive::OVERWRITE。

使用这两个参数,如果zip文件不存在,系统都会自动新建。

如果对zip文件对象操作成功,$zip->open这个方法会返回TRUE

*/

if ($zip->open('test.zip', ZipArchive::OVERWRITE) === TRUE)

{

$zip->addFile('image.txt');//假设加入的文件名是image.txt,在当前路径下

$zip->close();

}

三、文件追加内容添加到zip文件

$zip = new ZipArchive;

$res = $zip->open('test.zip', ZipArchive::CREATE);

if ($res === TRUE) {

$zip->addFromString('test.txt', 'file content goes here');

$zip->close();

echo 'ok';

} else {

echo 'failed';

}

四、将文件夹打包成zip文件

function addFileToZip($path, $zip) {

$handler = opendir($path); //打开当前文件夹由$path指定。

/*

循环的读取文件夹下的所有文件和文件夹

其中$filename = readdir($handler)是每次循环的时候将读取的文件名赋值给$filename,

为了不陷于死循环,所以还要让$filename !== false。

一定要用!==,因为如果某个文件名如果叫'0',或者某些被系统认为是代表false,用!=就会停止循环

*/

while (($filename = readdir($handler)) !== false) {

if ($filename != "." && $filename != "..") {//文件夹文件名字为'.'和‘..’,不要对他们进行操作

if (is_dir($path . "/" . $filename)) {// 如果读取的某个对象是文件夹,则递归

addFileToZip($path . "/" . $filename, $zip);

} else { //将文件加入zip对象

$zip->addFile($path . "/" . $filename);

}

}

}

@closedir($path);

}

$zip = new ZipArchive();

if ($zip->open('images.zip', ZipArchive::OVERWRITE) === TRUE) {

addFileToZip('images/', $zip); //调用方法,对要打包的根目录进行操作,并将ZipArchive的对象传递给方法

$zip->close(); //关闭处理的zip文件

}

五、ZipArchive方法如下:

ZipArchive::addFile — Adds a file to a ZIP archive from the given path

ZipArchive::addFromString — Add a file to a ZIP archive using its contents

ZipArchive::addGlob — Add files from a directory by glob pattern

ZipArchive::addPattern — Add files from a directory by PCRE pattern

ZipArchive::close — Close the active archive (opened or newly created)

ZipArchive::deleteIndex — delete an entry in the archive using its index

ZipArchive::deleteName — delete an entry in the archive using its name

ZipArchive::extractTo — Extract the archive contents

ZipArchive::getArchiveComment — Returns the Zip archive comment

ZipArchive::getCommentIndex — Returns the comment of an entry using the entry index

ZipArchive::getCommentName — Returns the comment of an entry using the entry name

ZipArchive::getExternalAttributesIndex — Retrieve the external attributes of an entry defined by its index

ZipArchive::getExternalAttributesName — Retrieve the external attributes of an entry defined by its name

ZipArchive::getFromIndex — Returns the entry contents using its index

ZipArchive::getFromName — Returns the entry contents using its name

ZipArchive::getNameIndex — Returns the name of an entry using its index

ZipArchive::getStatusString — Returns the status error message, system and/or zip messages

ZipArchive::getStream — Get a file handler to the entry defined by its name (read only).

ZipArchive::locateName — Returns the index of the entry in the archive

ZipArchive::open — Open a ZIP file archive

ZipArchive::renameIndex — Renames an entry defined by its index

ZipArchive::renameName — Renames an entry defined by its name

ZipArchive::setArchiveComment — Set the comment of a ZIP archive

ZipArchive::setCommentIndex — Set the comment of an entry defined by its index

ZipArchive::setCommentName — Set the comment of an entry defined by its name

ZipArchive::setExternalAttributesIndex — Set the external attributes of an entry defined by its index

ZipArchive::setExternalAttributesName — Set the external attributes of an entry defined by its name

ZipArchive::statIndex — Get the details of an entry defined by its index.

ZipArchive::statName — Get the details of an entry defined by its name.

ZipArchive::unchangeAll — Undo all changes done in the archive

ZipArchive::unchangeArchive — Revert all global changes done in the archive.

ZipArchive::unchangeIndex — Revert all changes done to an entry at the given index

ZipArchive::unchangeName — Revert all changes done to an entry with the given name.

另附一:几行代码实现PHP文件打包下载zip

/**

* 没有写成class 或者 function ,需要的朋友自己写,就这么几行。。

*/

$filename = "./test/test.zip"; //最终生成的文件名(含路径)

if(!file_exists($filename)){

//重新生成文件

$zip = new ZipArchive();//使用本类,linux需开启zlib,windows需取消php_zip.dll前的注释

if ($zip->open($filename, ZIPARCHIVE::CREATE)!==TRUE) {

exit('无法打开文件,或者文件创建失败');

}

foreach( $datalist as $val){

$attachfile = $attachmentDir . $val['filepath']; //获取原始文件路径

if(file_exists($attachfile)){

$zip->addFile( $attachfile , basename($attachfile));//第二个参数是放在压缩包中的文件名称,如果文件可能会有重复,就需要注意一下

}

}

$zip->close();//关闭

}

if(!file_exists($filename)){

exit("无法找到文件"); //即使创建,仍有可能失败。。。。

}

header("Cache-Control: public");

header("Content-Description: File Transfer");

header('Content-disposition: attachment; filename='.basename($filename)); //文件名

header("Content-Type: application/zip"); //zip格式的

header("Content-Transfer-Encoding: binary"); //告诉浏览器,这是二进制文件

header('Content-Length: '. filesize($filename)); //告诉浏览器,文件大小

@readfile($filename);

?>

另附二:文件打包,下载之使用PHP自带的ZipArchive压缩文件并下载打包好的文件

总结:

使用PHP下载文件的操作需要给出四个header(),可参考:PHP下载文件名中文乱码解决方法和PHP下载流程分析

计算文件的大小的时候,并不需要先打开文件,通过filesize($filename)就可以看出,如果需要先打开文件的话,filesize可能就会是这样的形式了filesize($filehandle)

向客户端回送数据的是,记得要设置一个buffer,用来指定每次向客户端输出多少数据,如:$buffer=1023。如果不指定的话,就会将整个文件全部写入内存当中,再一次性的讲数据传送给客户端

通过feof()函数,可以判断要读取的文件是否读完,如果还没读完,继续读取文件($file_data=fread()),并将数据回送给客户端(echo $file_data)

每次下载完成后,在客户端都会刷新下,说明了,其实每次都将数据写入到一个临时文件中,等全部下载完成后,再将所有的数据重新整合在一起

这里我使用的是绝对路径,绝对路径有个好处,就是适应性比较强,而且相对于相对路径,效率更高(免去了查找文件的过程)

分析下技术要点:

将文件打包成zip格式

下载文件的功能

要点解析:

这里我采用的是php自带的ZipArchive类

a) 我们只需要new一个ZipArchive对象,然后使用open方法创建一个zip文件,接着使用addFile方法,将要打包的文件写入刚刚创建的zip文件中,最好还得记得关闭该对象。

b) 注意点:使用open方法的时候,第二个参数$flags是可选的,$flags用来指定对打开的zip文件的处理方式,共有四种情况

i.     ZIPARCHIVE::OVERWRITE总是创建一个新的文件,如果指定的zip文件存在,则会覆盖掉

ii.    ZIPARCHIVE::CREATE     如果指定的zip文件不存在,则新建一个

iii.  ZIPARCHIVE::EXCL      如果指定的zip文件存在,则会报错

iv.  ZIPARCHIVE::CHECKCONS

下载文件的流程:

服务器端的工作:

-------------------------------------------

客户端的浏览器发送一个请求到处理下载的php文件。

注意:任何一个操作都首先需要写入到内存当中,不管是视频、音频还是文本文件,都需要先写入到内存当中。

换句话说,将“服务器”上的文件读入到“服务器”的内存当中的这个操作时必不可少的(注意:这里我将服务器三个字加上双引号,主要是说明这一系类的操作时在服务器上完成的)。

既然要将文件写入到内存当中,就必然要先将文件打开

所以这里就需要三个文件操作的函数了:

一:fopen($filename ,$mode)

二:fread ( int $handle , int $length )

三:fclose ( resource $handle )

---------------------------------------

客户端端的工作:

---------------------------------------

那么,如何将已经存在于服务器端内存当中的文件信息流,传给客户端呢?

答案是通过header()函数,客户端就知道该如何处理文件,是保存还是打开等等

最终的效果如下图所示:

require'./download.php';

/**

* 遍历目录,打包成zip格式

*/

class traverseDir{

public $currentdir;//当前目录

public $filename;//文件名

public $fileinfo;//用于保存当前目录下的所有文件名和目录名以及文件大小

public function __construct(){

$this->currentdir=getcwd();//返回当前目录

}

//遍历目录

public function scandir($filepath){

if (is_dir($filepath)){

$arr=scandir($filepath);

foreach ($arr as $k=>$v){

$this->fileinfo[$v][]=$this->getfilesize($v);

}

}else {

echo "";

}

}

/**

* 返回文件的大小

*

* @param string $filename 文件名

* @return 文件大小(KB)

*/

public function getfilesize($fname){

return filesize($fname)/1024;

}

/**

* 压缩文件(zip格式)

*/

public function tozip($items){

$zip=new ZipArchive();

$zipname=date('YmdHis',time());

if (!file_exists($zipname)){

$zip->open($zipname.'.zip',ZipArchive::OVERWRITE);//创建一个空的zip文件

for ($i=0;$i

$zip->addFile($this->currentdir.'/'.$items[$i],$items[$i]);

}

$zip->close();

$dw=new download($zipname.'.zip'); //下载文件

$dw->getfiles();

unlink($zipname.'.zip'); //下载完成后要进行删除

}

}

}

?>

/**

* 下载文件

*

*/

class download{

protected $_filename;

protected $_filepath;

protected $_filesize;//文件大小

public function __construct($filename){

$this->_filename=$filename;

$this->_filepath=dirname(__FILE__).'/'.$filename;

}

//获取文件名

public function getfilename(){

return $this->_filename;

}

//获取文件路径(包含文件名)

public function getfilepath(){

return $this->_filepath;

}

//获取文件大小

public function getfilesize(){

return $this->_filesize=number_format(filesize($this->_filepath)/(1024*1024),2);//去小数点后两位

}

//下载文件的功能

public function getfiles(){

//检查文件是否存在

if (file_exists($this->_filepath)){

//打开文件

$file = fopen($this->_filepath,"r");

//返回的文件类型

Header("Content-type: application/octet-stream");

//按照字节大小返回

Header("Accept-Ranges: bytes");

//返回文件的大小

Header("Accept-Length: ".filesize($this->_filepath));

//这里对客户端的弹出对话框,对应的文件名

Header("Content-Disposition: attachment; filename=".$this->_filename);

//修改之前,一次性将数据传输给客户端

echo fread($file, filesize($this->_filepath));

//修改之后,一次只传输1024个字节的数据给客户端

//向客户端回送数据

$buffer=1024;//

//判断文件是否读完

while (!feof($file)) {

//将文件读入内存

$file_data=fread($file,$buffer);

//每次向客户端回送1024个字节的数据

echo $file_data;

}

fclose($file);

}else {

echo "";

}

}

}

?> 页面显示的代码:

header("Content-type:text/html;charset=utf8");

require('./getfile.php');

$scandir=new traverseDir();

$scandir->scandir($scandir->currentdir);

$scandir->currentdir;

if (isset($_POST['down_load'])){

$items=$_POST['items'];

$scandir->tozip($items);//将文件压缩成zip格式

}

echo "当前的工作目录:".$scandir->currentdir;

echo "
当前目录下的所有文件";

?>

名称大小(KB)

$res=$scandir->fileinfo;

foreach ($res as $k=>$v){

if (!($k=='.' || $k=='..')) {//过滤掉.和..

?>

<?php echo $k; ?><?php echo number_format($v[0],0); ?>

}

}

?>

全选
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值