[TOC] # 文件系统概述 ### 文件的类型 在程序运行时,程序本身和数据一般都存在内存中,当程序运行结束后,存放在内存中的数据被释放。 如果需要长期保存程序运行所需的原始数据,或程序运行产生的结果,就必须以文件形式存储到外部存储介质上。 文件一般指存储在外部介质上具有名字(文件名)的一组相关数据集合。用文件可长期保存数据,并实现数据共享。 PHP是以UNIX的文件系统为模型的。因此在Windows系统中我们只能获得”file”、”dir”或者“unknown”三种文件类型。而在UNIX系统中,我们可以获得block、char、dir、fifo、file、link和unknown七种类型。 可以使用函数filetype()获取文件的具体类型。 ``` 语法:string filetype ( string filename ) ``` | 文件类型 | 描述 | | --- | --- | | block | 块设备文件,如某个磁盘分区,软驱,光驱CD-ROM等 | | char | 字符设备是指在I/O传输过程中以字符为单位进行传输的设备,如键盘、打印机等 | | dir | 目录类型,目录也是文件的一种 | | fifo | 命名管道,常用于将信息从一个进程传递到另一个进程 | | file | 普通文件类型,如文本文件或可执行文件等 | | link | 符号链接,是指向文件指针的指针。类似Windows中的快捷方式 | | unknow | 位置类型 | * `is_dir( )` -- 判断给定文件名是否是一个目录 语法结构:bool is_dir(名称) 返回类型:文件名存在并且是一个目录则返回 true,否则返回 false。 * `is_executable( )` -- 判断给定文件名是否可执行 语法结构:bool is_executable(名称) 返回类型:如果文件存在且可执行则返回 true ,否则返回 false 。 * `is_file( )` -- 判断给定文件名是否为一个正常的文件 语法结构:bool is_file(名称) 返回类型:如果文件存在且为正常的文件则返回 true 。 * `is_link( )` -- 判断给定文件名是否为一个符号连接 语法结构:bool is_link(名称) 返回类型:如果文件存在并且是一个符号连接则返回 true。 * `is_readable( )` -- 判断给定文件名是否可读 语法结构:bool is_readable(文件名称) 返回类型:如果文件存在并且可读则返回 true 。 * `is_writable( )` -- 判断给定的文件名是否可写 语法结构:bool is_writable(文件名称) 返回类型:如果文件存在并且可写则返回 true 。 ### 文件的属性 |函数名|作用|参数|返回值| |---|---|---|---| |file_exists( ) | 检查文件或目录是否存在| 文件名 |存在:true,不存在:false| |filesize() | 取得文件大小 |文件名 |返回大小字节数,出错:false| |is_readable() | 判断文件是否可读 | 文件名| 文件可读返回true| |is_writable() | 判断文件是否可写 | 文件名| 文件可写返回true| |is_executable()| 判断文件是否可执行 | 文件名| 文件可执行返回true| |filectime()| 获取文件的创建时间 | 文件名| 返回UNIX时间戳格式| |filemtime()| 获取文件的修改时间 | 文件名| 返回UNIX时间戳格式| |fileatime()| 获取文件的访问时间 | 文件名 |返回UNIX时间戳格式| |stat() | 获取文件大部分属性 | 文件名 |返回给定属性信息的数组| # 目录的基本操作 ### 解析目录路径 使用PHP脚本可以方便对目录进行操作,如创建目录、遍历目录、复值目录与删除目录等操作。 常用的文件目录路径格式: ``` $unixPath="/var/www/html/index.php"; //在UNIX系统中的绝对路径,必须使用"/"分隔 $winPath="C:\\Appserv\\www\\index.php"; //在Windows系统的绝对路径,默认使用"\"分隔 $winPath2="C:/Appserv/www/index.php"; //在Windows系统中也可使用“/”分隔。 ``` 注意使用绝对路径与相对路径。 * `basename` -- 返回路径中的文件名部分 ``` 语法:string basename ( string path [, string suffix] ) ``` 给出一个包含有指向一个文件的全路径的字符串,本函数返回基本的文件名。如果文件名是以 suffix 结束的,那这一部分也会被去掉。 * `dirname` -- 返回路径中的目录部分 ``` 语法:string dirname ( string path ) ``` 给出一个包含有指向一个文件的全路径的字符串,本函数返回去掉文件名后的目录名。 ``` <?php $path = "/home/httpd/html/index.php"; $file = basename($path); // $file值:"index.php" $file = basename($path, ".php "); // $file值:"index " $file = dirname($path); // $file值:"/home/httpd/html" ?> ``` * `pathinfo` -- 返回文件路径的信息 ``` 语法:array pathinfo ( string path [, int options] ) ``` pathinfo() 返回一个联合数组包含有 path 的信息。包括以下的数组单元:dirname,basename 和 extension。 * `realpath` -- 返回规范化的绝对路径名 ``` 语法:string realpath ( string path ) ``` realpath() 扩展所有的符号连接并且处理输入的 path 中的 '/./', '/../' 以及多余的 '/' 并返回规范化后的绝对路径名。返回的路径中没有符号连接,'/./' 或 '/../' 成分。 ``` <?php $path_parts = pathinfo("/www/htdocs/index.html"); echo $path_parts["dirname"]."\n"; // /www/htdocs echo $path_parts["basename"]."\n"; // index.html echo $path_parts["extension"]."\n"; // html ?> ``` ### 遍历目录 * `opendir` -- 打开目录句柄 ``` 语法:resource opendir ( string path [, resource context] ) ``` 打开一个目录句柄,可用于之后的 closedir(),readdir() 和 rewinddir() 调用中。 * `readdir` -- 从目录句柄中读取条目 ``` 语法:string readdir ( resource dir_handle ) ``` 返回当前目录指针位置的文件名,没有返回false,并将指针向下移动一位。文件名以在文件系统中的排序返回。 * `closedir` -- 关闭目录句柄 ``` 语法:void closedir ( resource dir_handle ) ``` 关闭由 dir_handle 指定的目录流。流必须之前被 opendir() 所打开。 * `rewinddir` -- 倒回目录句柄 ``` 语法:void rewinddir ( resource dir_handle ) ``` 将 dir_handle 指定的目录流重置到目录的开头。 ``` <?php //遍历指定目录实例 $num=0; //用来统计子目录和文件的个数 $dirname='phpMyAdmin'; //定义一个目录名 $dir_handle=opendir($dirname); //用opendir打开目录 //将遍历的目录和文件名使用表格格式输出 while($file=readdir($dir_handle)) { //使用readdir循环读取目录里的内容 $dirFile=$dirname."/".$file; //追加目录名 echo "文件名: ".$file." | "; //显示文件名 echo "大小: ".filesize($dirFile)." | "; //显示文件大小 echo "类型: ".filetype($dirFile)." | "; //显示文件类型 echo "创建时间: ".date("Y/n/t",filectime($dirFile))."<br/>"; $num++; } closedir($dir_handle); //关闭文件操作句柄 echo '在<b>'.$dirname.'</b>目录下共有文件<b>'.$num.'</b>个'; ?> ``` ### 统计目录大小 * `disk_free_space` -- 返回目录中的可用空间 ``` 语法:float disk_free_space ( string directory ) ``` 给出一个包含有一个目录的字符串,本函数将根据相应的文件系统或磁盘分区返回可用的字节数。 * `disk_total_space` -- 返回一个目录的磁盘总大小 ``` 语法:float disk_total_space ( string directory ) ``` 给出一个包含有一个目录的字符串,本函数将根据相应的文件系统或磁盘分区返回所有的字节数。 ``` <?php //定义一个统计指定目录大小的函数 function dirSize($dir){ $size=0; //1. 判断参数是否有效 if(!file_exists($dir) || !is_dir($dir)){ die("无效的参数!"); } //2. 打开目录 $dd = opendir($dir); //3. 跳过连个特殊目录:"."和".." readdir($dd);readdir($dd); //4. 开始遍历目录中文件 while($f = readdir($dd)){ //加上目录名 $file = rtrim($dir,"/")."/".$f; //判断是否是文件 if(is_file($file)){ $size+=filesize($file); } //判断是否是目录 if(is_dir($file)){ //采用递归调用,统计子目录大小 $size+=dirSize($file); } } //5. 关闭目录 closedir($dd); //6. 返回结果 return $size; } echo dirSize('test'); ?> ``` ### 建立与删除目录 * `mkdir` -- 新建目录 ``` 语法:bool mkdir (string pathname [,int mode]) ``` 尝试新建一个由 pathname 指定的目录。 * `rmdir` -- 删除目录 ``` 语法:bool rmdir ( string dirname ) ``` 尝试删除 dirname 所指定的目录。 该目录必须是空的,而且要有相应的权限。如果成功则返回 TRUE,失败则返回 FALSE。 * `unlink` -- 删除文件 ``` 语法:bool unlink ( string filename ) ``` 删除 filename。和 Unix C 的 unlink() 函数相似。如果成功则返回 TRUE,失败则返回 FALSE。 ``` <?php //自定义删除目录函数 function delDir($dir){ //1. 判断是否是有效文件或目录 if(!file_exists($dir)){ die("文件或目录{$dir}不存在!"); } //2. 若是文件就直接删除 if(is_file($dir)){ unlink($dir);return true; } //3. 打开目录 $dd = opendir($dir); //4. 跳过两个特殊目录:"." ".." readdir($dd);readdir($dd); //5. 遍历目录,并删除里面的文件 while($f = readdir($dd)){ //加上全路径 $file = rtrim($dir,"/")."/".$f; //判断是否是目录还是文件 if(is_dir($file)){ //采用递归调用删除子目录 delDir($file); }else{ //删除文件 unlink($file); } } //6. 关闭目录 closedir($dd); //7. 删除目录 rmdir($dir); return true; } ?> ``` ### 复制目录 * `copy` -- 拷贝文件 ``` 语法:bool copy ( string source, string dest ) ``` 将文件从 source 拷贝到 dest。如果成功则返回 TRUE,失败则返回 FALSE。 PHP中没有提供复制与移动目录的相关函数。如需要,只要自定义函数了。 如何定义一个目录复制函数呢? # 文件的基本操作 ### 文件的打开与关闭 * `fopen` -- 打开文件或者 URL ``` 语法:resource fopen ( string filename, string mode [, bool use_include_path [, resource zcontext]] ) ``` filename参数需要提供要被打开文件的URL。这个URL可以是脚本所在的服务器中的绝对路径,也可以是相对路径,还可以是网络资源用的文件。 mode 参数指定了所要求到该流的访问类型,(强烈建议附加b模式)。 如果也需要在 include_path中搜寻文件的话,可以将可选的第三个参数 use_include_path 设为 '1' 或 TRUE。 如果打开失败,本函数返回 FALSE。 * `fclose` -- 关闭一个已打开的文件指针 ``` 语法:bool fclose ( resource $handle ) ``` 将 handle 指向的文件关闭。 成功时返回 TRUE, 或者在失败时返回 FALSE. ``` <?php //使用绝对路径打开file.txt文件,选择只读模式,并返回资源$handle $handle = fopen("/home/rasmus/file.txt", "r"); //访问文档根目录下的文件,也以只读模式打开 $handle = fopen("{$_SERVER['DOCUMENT_ROOT']}/data/info.txt", "r"); //在 Windows 平台上,转义文件路径中的每个反斜线,或者用斜线,以二进制和只写模式组合 $handle = fopen("c:\\data\\file.gif", "wb"); //使用相对路径打开file.txt文件,选择只读模式,并返回资源$handle $handle = fopen("../data/info.txt", "r"); //打开远程文件, 使用HTTP协议只能以只读的模式打开 $handle = fopen("http://www.example.com/", "r"); //使用FTP协议打开远程文件,如果FTP服务器可写,则可以以写的模式打开 $handle = fopen("ftp://user:password@example.com/somefile.txt", "w"); ?> ``` ![](https://box.kancloud.cn/8d6024dd72b63920435ab920d70c1602_908x567.png) ### 写入文件 * `fwrite` -- 写入文件(可安全用于二进制文件) ``` 语法:int fwrite ( resource handle, string string [, int length] ) ``` fwrite() 把 string 的内容写入 文件指针 handle 处。 如果指定了 length,当写入了 length 个字节或者写完了 string 以后,写入就会停止,视乎先碰到哪种情况。 返回写入的字符数,出现错误时则返回 FALSE 。 ``` <?php //声明一个变量用来保存文件名 $fileName = "data.txt"; //使用fopen()函数以只写的模式打开文件,如果不存在则创建它, //打开失败则通过程序 $handle = fopen($fileName, 'w') or die('打开<b>'.$fileName.'</b>文件失败!!'); //循环10次写入10行数据到文件中 for($row=0; $row<10; $row++) { //写入文件 fwrite($handle, $row.": www.baidu.com\n"); } //关闭由fopen()打开的文件指针资源 fclose($handle); ?> ``` ### 读取文件内容 | 函数名 | 作用 | | --- | --- | | fread() | 读取文件(可安全用于二进制文件) | | file_get_contents() | 将文件读入字符串 | | fgets() | 从打开的文件中读取一行 | | fgetc() | 从打开的文件中读取一个字符 | | file() | 把文件读入一个数组中(无需使用fopen打开) | | readfile() | 读取一个文件,并输出到输出缓冲(无需使用fopen打开) | ``` <?php //从文件中读取指定字节数的内容存入到一个变量中 $filename = "data.txt"; //将本地文件名保存在变量中 $handle = fopen($filename, "r") or die("文件打开失败"); //以只读的方式打开文件 $contents = fread($handle, 100); //读取前100个字节;全部:filesize($filename) fclose($handle); //关闭文件资源 // echo $contents; //将从文件中读取的内容输出 //从文件中读取全部内容存入到一个变量中,每次读取一部分,循环读取 $handle = fopen ($filename, "rb") or die("文件打开失败"); //以只读的方式,模式加了’b’ $contents = ""; while (!feof($handle)) { //使用feof()判断文件结尾 $contents .= fread($handle, 1024); //每次读取1024个字节 } fclose($handle); //关闭文件资源 echo $contents; //将从文件中读取的全部内容输出 ?> ``` ``` <?php // echo file_get_contents("data.txt"); //读取文本文件中的内容并输出 $handle = fopen("data.txt", "r") or die("文件打开失败"); //以只读模式打开文件 while (!feof($handle)) { //循环读取第一行 $buffer = fgets($handle, 4096); //一次读取一行内容 echo $buffer."<br>"; //输出每一行 } fclose($handle); //关闭打开的文件资源 $fp = fopen('data.txt', 'r') or die("文件打开失败"); //以只读模式打开文件 while (false !== ($char = fgetc($fp))) { //在文件中每次循环读取一个字节符 echo $char."<br>"; //输出单个字符 } print_r( file("test.txt") ); //将文件test.txt中的内容读入到一个数组中,并输出 readfile("data.txt"); //直接将文件data.txt中的数据读出并输出到浏览器 ?> ``` ### 移动文件指针 * `ftell` -- 返回文件指针读/写的位置 ``` 语法:int ftell ( resource handle ) ``` 返回由 handle 指定的文件指针的位置,也就是文件流中的偏移量。 如果出错,返回 FALSE。文件指针必须是有效的,且必须指向一个通过 fopen()或 popen()成功打开的文件。 * `fseek` -- 在文件指针中定位 ``` 语法:int fseek ( resource handle,int offset [,int whence] ) ``` 在与 handle 关联的文件中设定文件指针位置。新位置,从文件头开始以字节数度量,是以 whence 指定的位置加上 offset。whence 的值定义为: SEEK_SET - 设定位置等于 offset 字节。 SEEK_CUR - 设定位置为当前位置加上 offset。 SEEK_END - 设定位置为文件尾加上 offset。(要移动到文件尾之前的位置,需要给 offset 传递一个负值。) 如果 没有指定 whence,默认为 SEEK_SET。 成功则返回 0;否则返回 -1。注意移动到 EOF 之后的位置不算错误 * `rewind` -- 倒回文件指针的位置 ``` 语法:bool rewind ( resource handle ) ``` 将 handle 的文件位置指针设为文件流的开头。 如果成功则返回 TRUE,失败则返回 FALSE。 文件指针必须合法,并且指向由 fopen() 成功打开的文件。 ### 文件的一些基本操作函数 * `copy` -- 拷贝文件 ``` 语法:bool copy ( string source, string dest ) ``` 将文件从 source 拷贝到 dest。如果成功则返回 TRUE,失败则返回 FALSE。 * `unlink` -- 删除文件 ``` 语法:bool unlink ( string filename ) ``` 删除 filename。和 Unix C 的 unlink() 函数相似。如果成功则返回 TRUE,失败则返 回 FALSE。 * `ftruncate` -- 将文件截断到给定的长度 ``` 语法:bool ftruncate ( resource handle, int size ) ``` 接受文件指针 handle 作为参数,并将文件大小截取为 size。如果成功则返回 TRUE,失败则返回 FALSE。 * `rename` -- 重命名一个文件或目录 ``` 语法:bool rename ( string oldname, string newname [, resource context] ) ``` 尝试把 oldname 重命名为 newname。 如果成功则返回 TRUE,失败则返回 FALSE。 > Have a try 1. 参考目录统计大小函数,自定义一个删除目录的函数,注意目录中的目录。 2. 使用文件处理函数,自定义一个目录复制函数。 3. 定义一个函数使用递归方式统计多维数组中的单元个数(递归函数练习)