【PHP】文件写入和读取详解

文章提纲:

一.实现文件读取和写入的基本思路

二.使用fopen方法打开文件

三.文件读取和文件写入操作

四.使用fclose方法关闭文件

五.文件指针的移动

六.Windows和UNIX下的回车和换行

一.实现文件读取和写入的基本思路:

1.通过fopen方法打开文件:$fp =fopen(/*参数,参数*/),fp为Resource类型

2.进行文件读取或者文件写入操作(这里使用的函数以1中返回的$fp作为参数)

3.   调用fclose($fp)关闭关闭文件

二:使用fopen方法打开文件

fopen(文件路径[string],打开模式[string])

<1>fopen的第一个参数为文件路径

写文件路径的方式:1绝对路径,2相对路径

1绝对路径:

在windows下工作的小伙伴们应该很熟悉,windows下的路径分隔符是“\”而不是“/”,但我们在写入路径时不能以钦定的“\”为分隔符

那如果我们以“\”分隔符写入路径会怎样呢?

<?php
     $fp = fopen("C:\wamp64\www\text.txt",‘w‘);
?>

运行后报错,提示路径参数无效

所以我们要把分隔符“\”换成“/”:

<?php
    $fp = fopen("C:/wamp64/www/text.txt",‘w‘);
?>

运行时无报错,说明参数是有效的。

【注意】fopen函数不能理解“\”分隔符,如果你想要使用“\”,那么要使用转义,如写成:"C:\\wamp64\\www\\text.txt"这种写法也是可以的,函数也能理解,不会报错。但即使这样,也不推荐使用“\”,因为在OS(mac)下只能识别“/”不能识别“\”

本小节的结论:推荐坚持使用“/”作为分隔符

2.相对路径:

上一小节介绍的是绝对路径的写法,但这样却带来了另外一个问题:服务器的目录结构可能会有较大的改变,这时原来写的绝对路径就要全部重写了,比如在我的电脑上的目标文件路径是C:/wamp64/www/text.txt,如果我把www文件夹改名为penghuwan呢?原来写入的路径参数就失效了。所以我们引入了相对路径的写法:

<?php
   $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
   $fp = fopen("$DOCUMENT_ROOT/text.txt",‘w‘);
?>

? $_SERVER是PHP的超级全局变量(在代码任何地方都可访问,类型是数组),通过$_SERVER[‘DOCUMENT_ROOT‘]可取到服务器的默认根目录

服务器的默认根目录可通过php.ini修改(这个可自行百度)

? $_SERVER[‘DOCUMENT_ROOT‘]在这里等同于C:/wamp64/www

本小节的结论:推荐使用相对路径

<2>fopen的第二个参数为打开模式

设置打开模式后,我们就相当于为接下来的读写操作设置了权限:

最基本的几个模式:

“r”:只能读取文件,不能写入文件(写入操作被忽略)

“w”:只能写入文件,不能读取文件(读取操作被忽略)

“a”:只追加文件,与“w”类似,区别是“w”删除原有的内容,“a”不删除原有内容,只追加内容

<?php
   $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
   $fp = fopen("$DOCUMENT_ROOT/text.txt",‘w‘);
   fwrite($fp,‘在写模式下写入‘);
   fclose($fp);
?>

在设置了写操作的权限后,就能正常地写入文件了

运行后打开C:/wamp64/www/text.txt:

这次我们把权限设置为只读,并尝试写入文本:‘在只读模式下写入‘

<?php
    $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
    $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);
    fwrite($fp,‘在读模式下写入‘);
    fclose($fp);
?>

运行后打开C:/wamp64/www/text.txt,发现文件内容并没有改变,说明由于没有设置相应的权限,操作被忽略了

关于打开模式的网络资料,我想大家最可能找到的是这张表:(图来自W3C)

很全面,但我觉得这张表对新手有些不太友好,让人看后不知多云。 r是只读,w是只写(原来有的内容全删除),a是追加(不删除原有内容),这都好理解。

但r+,w+,和a+的区别和联系讲的实在太模糊了呀。 这里我就想详细地讲一下r+,w+,和a+三者的区别和联系:

首先r+,w+,和a+都是可读可写的,读取时的方式是一样的,关键在于写入方式的不同:

r+: 从文件[头部][覆盖]原有内容 ([不删除]原有内容);

a+:从文件[尾部][追加]内容 ([不删除]原有内容);

w+:[完全删除]原有内容,然后[再添加]新的内容

下面我依次演示上述的结论,首先我们没有写入的时候文本是”I am initialized value”(意为我是初始值)

? 采用r+模式写入文本“r+ mode”

<?php
    $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
    $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r+‘);
    fwrite($fp,‘r+ mode‘);
    fclose($fp);
?>

运行后再打开文本,发现“I am in”被“r+ mode”覆盖了:

? 采用a+模式写入文本“a+ mode”

基于”I am initialized value”的初始文本我们运行以下代码:

<?php
    $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
    $fp = fopen("$DOCUMENT_ROOT/text.txt",‘a+‘);
    fwrite($fp,‘a+ mode‘);
    fclose($fp);
?>

I am initialized value没有被删除和覆盖,而是在后面追加了a+ mode的这一段新文本

运行多次后:

?采用w+模式写入文本“w+ mode”

基于”I am initialized value”的初始文本我们运行以下代码:

<?php
   $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
   $fp = fopen("$DOCUMENT_ROOT/text.txt",‘w+‘);
   fwrite($fp,‘w+ mode‘);
   fclose($fp);
?>

运行后,我们发现”I am initialized value”已经被删除了,然后才加上了“w+ mode”这段新文本

【注意】r+,a+,w+还有一个区别是a+,w+在文件不存在时则创建文件,r+文件不存在时报错

【吐槽】:关于r+和w+,a+的区别,我找了网络上,包括W3C和各种博客文章以及那本“PHP圣经”上的各种资料,发现都是一笔带过去的,这也是我写这篇文章的原因

三.文件读取和文件写入操作

先说说几个比较重要的函数:

? file_exists():判断文件是否存在,返回布尔值

? filesize():判断一个文件大小,返回文件的字节数,为整型数字

? unlink():删除一个文件

写入文件

fwrite(资源文件对象[string],写入方式[string]),资源文件对象即为fopen方法返回的参数,为Resource类型,写入方式可以是w(或者w+,a+,r+)

已经有上面的例子,这里就不放demo了

读取文件

这是我们要读取的文件内容:

读取文件的方式有以下几种:

1.一次读取一个字节的数据 fgetc()

2.一次读取指定的字节数的数据 fread()

3.一次读取一行数据 fgets()/fgetcsv()

4.一次读完全部数据  fpassthru()/ file()

1. 一次读取一个字节 —— 通过fgetc()获取单个字节

<?php
     $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
     $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);//打开文件
     if(file_exists("$DOCUMENT_ROOT/text.txt")){//当文件存在时,才读取内容
          while(!feof($fp)){//判断文件指针是否到达末尾
               $c = fgetc($fp);//每执行一次fgetc(),文件指针就向后移动一位
               echo $c;//输出获取到的字节
            }
       }
     fclose($fp);//关闭文件
?>

运行:

【注意】:无论是按文本格式输入输出还是按二进制格式输出,fgetc()每次获取的是一个字节而不是一个字符

上面的例子中我们是逐个输出,现在让我们只做一次输出,看看结果怎样:

<?php
    $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
    $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);
    echo fgetc($fp);//只做一次输出
    close($fp);
?>

运行结果如下,我们得到的不是汉字“我”,而是一个乱码,其实这个乱码就是一个字节

<?php
     $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
     $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);
     echo fgetc($fp);//连续做三次输出
     echo fgetc($fp);
     echo fgetc($fp);
     fclose($fp);
?>

2.一次读取多个字节 ——通过fread()方法:

<?php
   $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
   $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);
   echo fread($fp, 3);//一次输出三个字节即一个汉字字符(UTF-8)
   fclose($fp);
?>

运行结果:

改成:

echo fread($fp, 6);

运行结果如下,输出了6个字节也即两个汉字字符(UTF-8)

3.一次读取一行——通过fgets()获取一行内容

<?php
       $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘]
       $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);//打开文件
       if(file_exists("$DOCUMENT_ROOT/text.txt")){//当文件存在时,才读取内容
          while(!feof($fp)){//判断文件指针是否到达末尾
             $line = fgets($fp);//返回一行文本,并将文件指针移动到下一行头部
             echo $line."<br/>";//输出获取到的一行文本
          }
       }
       fclose($fp);//关闭文件
?>

fgets()其实还有第二个参数,这个参数规定了每一行能读取的最大字节数(注意是字节数不是字符数):

【注意】在UTF-8编码下汉字3字节,字母1字节

下面我修改上面的一行,代码,使获取的每一行最大字符数为3(也即字节数为9)

$line = fgets($fp,10);

Demo:

【注意】:这里我fgets()里第二个参数为10,为什么是10呢?因为

1.这里的长度是按字节数算的

2.一个汉字占3个字节。fgets($fp,10)代表一次最多读取10 - 1 = 9字节

4.一次读完全部文件 ——fpassthru() or file()?

fpassthru()将读取文件并直接输出(无处理过程)

<?php
      $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
      $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);
      fpassthru($fp);
      fclose($fp);
?>

运行结果:

【注意】这里需要注意一点的是,我们并没有从fpassthru($fp)获取到返回值然后echo到页面上去,也就是说这个方法是会强制输出获取的内容的,而并不是像之前例子的方法那样返回文本,允许我们保存到变量中才将其输出

将读取到的全部内容保存到一个数组中,每个数组元素为一行的内容——fille()

<?php
   $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
   $file_array = file("$DOCUMENT_ROOT/text.txt");//取到文件数组
   foreach ($file_array as $value) {//输出数组元素
       echo $value."<br/>";
    }
?>

【注意】:这里我们并不需要写fopen和fclose哦!也就是说file()方法已经帮我们做了这一步了

四.使用fclose方法关闭文件

fclose()将返回一个布尔值,成功关闭为true,关闭失败为false(失败的情况很少出现,可不考虑)

是否打开文件后一定要关闭?

1即使不手写fclose,在PHP脚本执行结束后,也会自动关闭文件的

2但在一个长时间执行的脚本中,如果不写关闭文件的fclose(),在文件加锁的情况下会造成操作的阻塞,所以,写fclose是个好习惯

五.文件指针的移动

我们上面调用的读取文件的函数,其实都是基于文件指针去打印的,每读取一段字节内容,文件指针就向后移动一段字节长度,直到被读取的文件最大字节长度为止

<?php
          $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
          function print_file_pointer($fp){//定义一个打印文件指针位置的函数
              echo " <br/>//此时文件指针的位置:";
              echo ftell($fp)."<br/>";
          }
          $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);
          echo fgetc($fp);//通过fgetc连续输出三个字节
          echo fgetc($fp);
          echo fgetc($fp);
          print_file_pointer($fp);//打印此刻文件指针的位置

          echo fread($fp,6);//通过fread一次输出6字节
          print_file_pointer($fp);//打印此刻文件指针的位置

          echo fgets($fp); //通过fgets输出一整行
          print_file_pointer($fp);//打印此刻文件指针的位置

          fpassthru($fp); //一次性输出全部内容
          print_file_pointer($fp);//打印此刻文件指针的位置

          fseek($fp, 33);//使文件指针移动到33字节位置
          print_file_pointer($fp);//打印此刻文件指针的位置

          rewind($fp);//使文件指针移动到0字节位置(初始位置)
          print_file_pointer($fp);//打印此刻文件指针的位置
$fclose($fp);
?>

Demo:

所以我们需要正确理解fgets(),fpassthru()这些函数的作用:

fgets():从当前文件指针的位置到本行结束的数据,而不是一定输出一整行

fpassthru():从当前文件指针的位置到全部内容结束的数据,而不是一定输出所有的数据

但在这里你可能会有疑问:为什么输出“湖湾”后的指针位置会是17而不是15呢?按理说输出“我叫彭湖湾”这5个汉字一共占3*5  = 15个字节,多出来的17 - 15 =2字节是什么呢?

多出来的两个字节是windows下的回车换行符\n\r

\n是换行,占一字节,\r是回车,占一字节,在六中我将会介绍

六.Windows和UNIX下的回车和换行

<?php
      $DOCUMENT_ROOT = $_SERVER[‘DOCUMENT_ROOT‘];
      $fp = fopen("$DOCUMENT_ROOT/text.txt",‘r‘);
      while(!feof($fp)){
        echo fgets($fp);
        echo ftell($fp);
      }
      fclose($fp);
?>

我们在windows下敲下回车键的时候,相当于键入了\n\r,所以“我叫彭湖湾”的15字节+“\n\r”的2字节 = 17字节

在mac下不一样的是:敲下回车键的时候,相当于只键入了\n,所以“我叫彭湖湾”的15字节+“\n”的1字节 = 16字节

【完】

时间: 2024-11-05 12:15:05

【PHP】文件写入和读取详解的相关文章

Java研究之文件路径的读取详解

 记得在操作系统中了解到文件读取有两种方式,当然这在各编程语言中也是通用的,所以java路径也分,相对和绝对路径.上章我们分享了Java研究之学习设计模式-组合模式详解有兴趣的朋友可以去看下. 绝对路径 绝对路径URI ,听着和URL很相似,那我们就来看看吧. URI(Uniformresource Identifier)统一资源标示符.URL统一资源定位符,是一个定位器,还说明了具体如何找到资源.所以他们就有一种抽象和继承的关系.URI抽象的说明了统一资源表示符号,而URL是具体的标识符的

VMware虚拟机文件(后缀)详解

VMware虚拟机文件(后缀)详解 虚拟机的文件管理由VMware Workstation来执行,一个虚拟机一般以一系列文件的形式储存在宿主机中,这些文件一般在由workstation为虚拟机所创建的那个目录中.这里列出了这些关键文件的扩展名.在这些例子中,<vmname>表示你的虚拟机名字. .log <vm name>.log or vmware.log这个文件记录了VMware Workstation对虚拟机调节运行的情况.当你碰到问题时,这些文件对我们做出故障诊断非常有用.

Linux文件或目录权限详解

版权声明: ########################################################################### 本文的所有内容均来自作者刘春凯的学习总结,未经本人许可,禁止私自转发及使用. QQ:1151887353 E-mail:[email protected] [email protected] ########################################################################

Linux系统根文件以及命名规则详解

一.Linux系统根文件详解 Linux的重要哲学思想其实就是:将程序的配置文件保存为纯文本格式. 1./boot:系统启动文件,如:内核文件,iniyrd以及gurb(bootloarder) 2./dev:目录下为设备文件,设备文件又分为块设备和字符设备: 块设备:按数据块随机访问,没有顺序. 字符设备:线性访问,按字符为单位进行. 注:其中背景为黑色,字体为***的文件,为特殊文件,"1,   0"分别为文件的主设备号和次设备号 [[email protected] ~]# ls

linux下文件打包、压缩详解

Linux平台下,有如下几种常见的压缩工具: ========================================================================= 工 具 文件扩展名 描述 ------------------------------------------------------------------------- bzip2 .bz2 采用Burrows-Wheeler块排序文本压缩算法和霍夫曼编码 compress .Z 原始的Unix文件压

ln命令总结,软链接&硬链接&文件删除原理画图详解

ln命令总结,软链接&硬链接&文件删除原理画图详解

MFC中文件对话框类CFileDialog详解及文件过滤器说明

当前位置?:?首页???文章分类?:? 开发 ??? MFC中文件对话框类CFileDialog详解及文件过滤器说明 上一篇 利用OpenCV从摄像头获得图像的坐标原点是在左下角 下一篇 Word中为图表生成目录 域名迁移公告 2017年12月20日起,本博客迁移到新域名madaimeng.com,旧域名masikkk.com不再更新内容,但将永久保持可访问! 阅读 大专栏  MFC中文件对话框类CFileDialog详解及文件过滤器说明busuanzi_value_page_pv"> 评

linux下find(文件查找)命令的详解

文件查找命令locate和find详解 locate 配合数据库缓存,快速查看文件位置,非实时查找( 数据库查找) find 实际搜寻硬盘查询文件名称 ,实时查找 locate简介 locate命令其实是find -name的另一种写法,但是要比后者快得多,原因在于它不搜索具体目录,而是搜索一个数据库/var/lib/locatedb,这个数据库中含有本地所有文件信息.Linux系统自动创建这个数据库,并且每天自动更新一次,所以使用locate命令查不到最新变动过的文件.为了避免这种情况,可以在

sas数据读取详解 四种读取数据方式以及数据指针的位置 、读取mess data的两个小工具、特殊的读取技巧、infile语句及其选项(dsd dlm missover truncover obs firstobs)、proc import、自定义缺失值

(The record length is the number of characters, including spaces, in a data line.) If your data lines are long, and it looks like SAS is not reading all your data, then use the LRECL= option in the INFILE statement to specify a record length at least