0x01 文件系统概述
1.1文件类型
在程序运行时,程序本身和数据一般都存在内存中,当程序运行结束后,存放在内存中的数据被释放.
如果需要长期保存程序运行所需的原始数据,或程序运行产生的结果,就必须以文件形式存储到外部存储介质上.
文件一般指存储在外部介质上具有名字(文件名)的一组相关数据集合。用文件可长期保存数据,并实现数据共享。
PHP是以UNIX的文件系统为模型的。因此在Windows系统中我们只能获得”file”、”dir”或者“unknown”三种文件类型。而在UNIX系统中,我们可以获得block、char、dir、fifo、file、link和unknown七种类型。
可以使用函数filetype()获取文件的具体类型。
比如这里我们想获取当前路径下2.png的文件类型
<?php
var_dump(filetype("./2.png"));
?>
输出string(4) "file"
UNIX系统中7种文件类型说明
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 。
1.2文件的属性
<?php
var_dump(date("Y-m-d H:i:s",filectime("./2.png")));
?>
返回结果string(19) "2020-09-22 04:07:13",其他几个类似
0x02 目录的基本操作
2.1 解析目录路径
使用PHP脚本可以方便对目录进行操作,如创建目录、遍历目录、复值目录与删除目录等操作。
常用的文件目录路径格式:
$unixPath="/var/www/html/index.php";
在UNIX系统中的绝对路径,必须使用"/"分隔
$winPath="C:Appservwwwindex.php";
在Windows系统的绝对路径,默认使用""分隔
$winPath2="C:/Appserv/www/index.php";
在Windows系统中也可使用“/”分隔。
注意使用绝对路径与相对路径。
PHP文件路径相关函数
basename -- 返回路径中的文件名部分
语法:string basename ( string path [, string suffix] )
给出一个包含有指向一个文件的全路径的字符串,本函数返回基本的文件名。如果文件名是以 suffix 结束的,那这一部分也会被去掉。
dirname -- 返回路径中的目录部分
语法:string dirname ( string path )
给出一个包含有指向一个文件的全路径的字符串,本函数返回去掉文件名后的目录名。
<?php
$path = "G:/PhpStudy20180211/PHPTutorial/WWW/m/2.png";
$file = basename($path);echo $file;echo "<br>";
$file = basename($path,"png");echo $file;echo "<br>";
$file = dirname($path);echo $file;
?>
输出
2.png
2.
G:/PhpStudy20180211/PHPTutorial/WWW/m
这里我们可以只取文件
<?php
$path = isset($_GET['path']) ? $_GET['path'] : '';
var_dump(basename($path));
?>
这样url输入多少个越级都不会有反应http://127.0.0.1/m/index.php?path=../../../index.php输出index.php,但是再linux里面可以通过......去绕过
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"]."
"; // /www/htdocs
echo $path_parts["basename"]."
"; // index.html
echo $path_parts["extension"]."
"; // html
?>
2.2 遍历目录
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 //遍历指定目录实例
header("Content-type: text/html; charset=utf-8");
$num=0; //用来统计子目录和文件的个数
$dirname='./'; //定义一个目录名
$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>个';
输出
文件名: . | 大小: 0 | 类型: dir | 创建时间: 2020/9/30
文件名: .. | 大小: 16384 | 类型: dir | 创建时间: 2019/9/30
文件名: 2.png | 大小: 955 | 类型: file | 创建时间: 2020/9/30
文件名: index.php | 大小: 956 | 类型: file | 创建时间: 2020/9/30
在./目录下共有文件4个
2.3 统计目录大小
<?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("./");
?>
可以除以1024就是实际大小kb
2.4 建立与删除目录
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。
如果想要循环删除其实就可以用到上面的代码$size+=dirSize改为unlink就可以了
2.5 复制目录
copy -- 拷贝文件
语法:bool copy ( string source, string dest )
将文件从 source 拷贝到 dest。如果成功则返回 TRUE,失败则返回 FALSE。
PHP中没有提供复制与移动目录的相关函数。如需要,只要自定义函数了。
<?php
copy("./2.png", "./1.jpg");
?>
0x03 文件的基本操作
3.1 文件的打开与关闭
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");
3.2 写入文件
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.": https://www.cnblogs.com/yicunyiye/
");
}
//关闭由fopen()打开的文件指针资源
fclose($handle);
?>
data.txt
0: https://www.cnblogs.com/yicunyiye/
1: https://www.cnblogs.com/yicunyiye/
2: https://www.cnblogs.com/yicunyiye/
3: https://www.cnblogs.com/yicunyiye/
4: https://www.cnblogs.com/yicunyiye/
5: https://www.cnblogs.com/yicunyiye/
6: https://www.cnblogs.com/yicunyiye/
7: https://www.cnblogs.com/yicunyiye/
8: https://www.cnblogs.com/yicunyiye/
9: https://www.cnblogs.com/yicunyiye/
3.3 读取文件内容
<?php
//从文件中读取指定字节数的内容存入到一个变量中
$filename = "data.txt"; //将本地文件名保存在变量中
$handle = fopen($filename, "r") or die("文件打开失败"); //以只读的方式打开文件
$contents = fread($handle, 100); //读取前100个字节;全部:filesize($filename)
fclose($handle); //关闭文件资源
echo $contents; //将从文件中读取的内容输出
?>
输出0: https://www.cnblogs.com/yicunyiye/ 1: https://www.cnblogs.com/yicunyiye/ 2: https://www.cnblogs.c
但是一般使用file_put_contents和file_get_contents多一些,因为比较简单
<?php
file_put_contents("./data.txt", "111");
echo file_get_contents("./data.txt");
?>
这个函数还可以追加
file_put_contents("./data.txt", "追加",FILE_APPEND);
3.4 访问远程文件
如果需要访问远程文件,必须在PHP的配置文件中激活“allow_url_fopen”选项,才能使用fopen( )函数打开运程文件。而且还要确定其他服务器中的文件是否访问权限,如果使用PHP协议对远程文件进行链接,只能以“只读”模式打开。如果需要访问的远程FTP服务器中,对所提供的用户开启了“可写”权限,则使用FTP协议链接远程的文件时,就可以使用“只写”或“只读”模式打开文件。但不可以使用“可读可写”的模式。
使用PHP访问远程文件就像访问本地文件一样,都是使用相同的读写函数处理。
$file=fopen(“http://www.111/”,”r”) or die(“打开远程文件失败!!”);
$file=fopen(“ftp://user:password@ftp.111.net/path/to/file”,”w”);
3.5 移动文件指针
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() 成功打开的文件。
3.6 文件的锁定机制
flock -- 轻便的咨询文件锁定
语法: bool flock ( int handle, int operation [, int &wouldblock] )
PHP 支持以咨询方式(也就是说所有访问程序必须使用同一方式锁定, 否则它不会工作)锁定全部文件的一种轻便方法。
handle 必须是一个已经打开的文件指针。
operation 可以是以下值之一:
要取得共享锁定(读取程序),将 operation 设为 LOCK_SH。
要取得独占锁定(写入程序),将 operation 设为 LOCK_EX。
要释放锁定(无论共享或独占),将 operation 设为 LOCK_UN。
如果你不希望 flock() 在锁定时堵塞,则给 operation 加上 LOCK_NB。
如果成功则返回 TRUE,失败则返回 FALSE。
3.7 文件的一些基本操作函数
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。
0x04 文件上传
注意几个特征属性:
POST方法:
表单最常用的功能,向目标页面传递变量,我们在上传文件的时候,会在表单中设置相应的属性,来完成文件的传递
enctype="multipart/form-data"
这样服务器就会知道,我们要传递一个文件,这样服务器可以知道上载的文件带有常规的表单信息。
MAX_FILE_SIZE
此字段必须在文件输入字段之前,控制最大的传递文件的大小(字节)
<input type="file" name="userfile">
设置浏览器文件输入浏览按钮
upload.html
<html>
<head><title>文件上传</title></head>
<body>
<form action="index.php" method="post" enctype="multipart/form-data">
<input type="hidden" name="MAX_FILE_SIZE" value="1000000">
选择文件:<input type="file" name="myfile">
<input type="submit" value="上传文件">
</form>
</body>
</html>
PHP配置文件中与文件上传有关的选项
$_FILES多维数组
超级全局数组$_FILES
1、$_FILES["myfile"]["name"]中的值是:
客户端文件系统的文件的名称
2、$_FILES["myfile"]["type"]中的值是:
客户端传递的文件的类型
3、$_FILES["myfile"]["size"]中的值是:
文件的字节的大小
4、$_FILES["myfile"]["tmp_name"]中的值是:
文件被上传后在服务器存储的临时全路径
5、$_FILES["myfile"]["error"]中的值是:
文件上传的错误代码-php 4.2以后增加的功能
比如说这里index.php为
<?php
var_dump($_FILES);
?>
我们上传一个1.doc,返回为
array(1) { ["myfile"]=> array(5) { ["name"]=> string(6) "1.docx" ["type"]=> string(71) "application/vnd.openxmlformats-officedocument.wordprocessingml.document" ["tmp_name"]=> string(53) "C:UsersAdministratorAppDataLocalTempphpD841.tmp" ["error"]=> int(0) ["size"]=> int(917105) } }
这里可以看到error对应值为0
存储在$_FILES["myfile"]["error"]中的值
伴随文件上传时产生的错误信息代码是在PHP4.2.0版本中引入的,具体如下:
值为0:表示没有发生任何错误。
值为1:表示上传文件的大小超出了约定值。文件大小的最大值是在PHP配置文件中指定的,该指令是:upload_max_filesize。
值为2:表示上传文件大小超出了HTML表单隐藏域属性的MAX_FILE_SIZE元素所指定的最大值。
值为3:表示文件只被部分上传。
值为4:表示没有上传任何文件。
值为6:表示找不到临时文件夹。PHP 4.3.10 和 PHP 5.0.3 引进。
值为7:表示文件写入失败。PHP 5.1.0 引进。
常见数据格式(MIME)
我们其实上传到我们的本地也很简单,比如刚刚上传的1.docx
<?php
move_uploaded_file($_FILES['myfile']['tmp_name'], $_FILES['myfile']['name']);
?>
0x05 文件下载
<?php
<?php
$filename=".2.png";
$basename=pathinfo($filename);
header("Content-Type: image/png"); //指定下载文件类型的
header("Content-Disposition:attachment;filename=".$basename["basename"]);
//指定下载文件的描述信息
header("Content-Length:".filesize($filename)); //指定文件大小的
readfile($filename);//将内容输出,以便下载。