溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

PHP如何實(shí)現(xiàn)文件寫入和讀取

發(fā)布時(shí)間:2021-07-02 17:07:26 來源:億速云 閱讀:181 作者:chen 欄目:開發(fā)技術(shù)

本篇內(nèi)容主要講解“PHP如何實(shí)現(xiàn)文件寫入和讀取”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實(shí)用性強(qiáng)。下面就讓小編來帶大家學(xué)習(xí)“PHP如何實(shí)現(xiàn)文件寫入和讀取”吧!

文章提綱:

一.實(shí)現(xiàn)文件讀取和寫入的基本思路

二.使用fopen方法打開文件

三.文件讀取和文件寫入操作

四.使用fclose方法關(guān)閉文件

五.文件指針的移動(dòng)

六.Windows和UNIX下的回車和換行

一.實(shí)現(xiàn)文件讀取和寫入的基本思路:

1.通過fopen方法打開文件:$fp =fopen(/*參數(shù),參數(shù)*/),fp為Resource類型
2.進(jìn)行文件讀取或者文件寫入操作(這里使用的函數(shù)以1中返回的$fp作為參數(shù))
3. 調(diào)用fclose($fp)關(guān)閉關(guān)閉文件

二:使用fopen方法打開文件

fopen(文件路徑[string],打開模式[string])

<1>fopen的第一個(gè)參數(shù)為文件路徑
寫文件路徑的方式:1絕對(duì)路徑,2相對(duì)路徑

1絕對(duì)路徑:

在windows下工作的小伙伴們應(yīng)該很熟悉,windows下的路徑分隔符是“\”而不是“/”,但我們?cè)趯懭肼窂綍r(shí)不能以欽定的“\”為分隔符

PHP如何實(shí)現(xiàn)文件寫入和讀取

那如果我們以“\”分隔符寫入路徑會(huì)怎樣呢?

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

運(yùn)行后報(bào)錯(cuò),提示路徑參數(shù)無效

PHP如何實(shí)現(xiàn)文件寫入和讀取

所以我們要把分隔符“\”換成“/”:

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

運(yùn)行時(shí)無報(bào)錯(cuò),說明參數(shù)是有效的。

【注意】fopen函數(shù)不能理解“\”分隔符,如果你想要使用“\”,那么要使用轉(zhuǎn)義,如寫成:"C:\\wamp64\\www\\text.txt"這種寫法也是可以的,函數(shù)也能理解,不會(huì)報(bào)錯(cuò)。但即使這樣,也不推薦使用“\”,因?yàn)樵贠S(mac)下只能識(shí)別“/”不能識(shí)別“\”

本小節(jié)的結(jié)論:推薦堅(jiān)持使用“/”作為分隔符

2.相對(duì)路徑:

上一小節(jié)介紹的是絕對(duì)路徑的寫法,但這樣卻帶來了另外一個(gè)問題:服務(wù)器的目錄結(jié)構(gòu)可能會(huì)有較大的改變,這時(shí)原來寫的絕對(duì)路徑就要全部重寫了,比如在我的電腦上的目標(biāo)文件路徑是C:/wamp64/www/text.txt,如果我把www文件夾改名為penghuwan呢?原來寫入的路徑參數(shù)就失效了。所以我們引入了相對(duì)路徑的寫法:

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

? $_SERVER是PHP的超級(jí)全局變量(在代碼任何地方都可訪問,類型是數(shù)組),通過$_SERVER['DOCUMENT_ROOT']可取到服務(wù)器的默認(rèn)根目錄

服務(wù)器的默認(rèn)根目錄可通過php.ini修改(這個(gè)可自行百度)

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

本小節(jié)的結(jié)論:推薦使用相對(duì)路徑

<2>fopen的第二個(gè)參數(shù)為打開模式

設(shè)置打開模式后,我們就相當(dāng)于為接下來的讀寫操作設(shè)置了權(quán)限:

最基本的幾個(gè)模式:

“r”:只能讀取文件,不能寫入文件(寫入操作被忽略)
“w”:只能寫入文件,不能讀取文件(讀取操作被忽略)
“a”:只追加文件,與“w”類似,區(qū)別是“w”刪除原有的內(nèi)容,“a”不刪除原有內(nèi)容,只追加內(nèi)容

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

在設(shè)置了寫操作的權(quán)限后,就能正常地寫入文件了

運(yùn)行后打開C:/wamp64/www/text.txt:

PHP如何實(shí)現(xiàn)文件寫入和讀取

這次我們把權(quán)限設(shè)置為只讀,并嘗試寫入文本:'在只讀模式下寫入'

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

運(yùn)行后打開C:/wamp64/www/text.txt,發(fā)現(xiàn)文件內(nèi)容并沒有改變,說明由于沒有設(shè)置相應(yīng)的權(quán)限,操作被忽略了

PHP如何實(shí)現(xiàn)文件寫入和讀取

關(guān)于打開模式的網(wǎng)絡(luò)資料,我想大家最可能找到的是這張表:(圖來自W3C)

PHP如何實(shí)現(xiàn)文件寫入和讀取

很全面,但我覺得這張表對(duì)新手有些不太友好,讓人看后不知多云。 r是只讀,w是只寫(原來有的內(nèi)容全刪除),a是追加(不刪除原有內(nèi)容),這都好理解。

但r+,w+,和a+的區(qū)別和聯(lián)系講的實(shí)在太模糊了呀。 這里我就想詳細(xì)地講一下r+,w+,和a+三者的區(qū)別和聯(lián)系:

首先r+,w+,和a+都是可讀可寫的,讀取時(shí)的方式是一樣的,關(guān)鍵在于寫入方式的不同:

r+: 從文件[頭部][覆蓋]原有內(nèi)容 ([不刪除]原有內(nèi)容);

a+:從文件[尾部][追加]內(nèi)容 ([不刪除]原有內(nèi)容);

w+:[完全刪除]原有內(nèi)容,然后[再添加]新的內(nèi)容

下面我依次演示上述的結(jié)論,首先我們沒有寫入的時(shí)候文本是”I am initialized value”(意為我是初始值)

PHP如何實(shí)現(xiàn)文件寫入和讀取

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

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

運(yùn)行后再打開文本,發(fā)現(xiàn)“I am in”被“r+ mode”覆蓋了:

PHP如何實(shí)現(xiàn)文件寫入和讀取

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

基于”I am initialized value”的初始文本我們運(yùn)行以下代碼:

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

PHP如何實(shí)現(xiàn)文件寫入和讀取

I am initialized value沒有被刪除和覆蓋,而是在后面追加了a+ mode的這一段新文本

運(yùn)行多次后:

PHP如何實(shí)現(xiàn)文件寫入和讀取

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

基于”I am initialized value”的初始文本我們運(yùn)行以下代碼:

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

運(yùn)行后,我們發(fā)現(xiàn)”I am initialized value”已經(jīng)被刪除了,然后才加上了“w+ mode”這段新文本

PHP如何實(shí)現(xiàn)文件寫入和讀取

【注意】r+,a+,w+還有一個(gè)區(qū)別是a+,w+在文件不存在時(shí)則創(chuàng)建文件,r+文件不存在時(shí)報(bào)錯(cuò)

【吐槽】:關(guān)于r+和w+,a+的區(qū)別,我找了網(wǎng)絡(luò)上,包括W3C和各種博客文章以及那本“PHP圣經(jīng)”上的各種資料,發(fā)現(xiàn)都是一筆帶過去的,這也是我寫這篇文章的原因

三.文件讀取和文件寫入操作

先說說幾個(gè)比較重要的函數(shù):

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

? filesize():判斷一個(gè)文件大小,返回文件的字節(jié)數(shù),為整型數(shù)字

? unlink():刪除一個(gè)文件

寫入文件

fwrite(資源文件對(duì)象[string],寫入方式[string]),資源文件對(duì)象即為fopen方法返回的參數(shù),為Resource類型,寫入方式可以是w(或者w+,a+,r+)

已經(jīng)有上面的例子,這里就不放demo了

讀取文件

這是我們要讀取的文件內(nèi)容:

PHP如何實(shí)現(xiàn)文件寫入和讀取

讀取文件的方式有以下幾種:

1.一次讀取一個(gè)字節(jié)的數(shù)據(jù) fgetc()

2.一次讀取指定的字節(jié)數(shù)的數(shù)據(jù) fread()

3.一次讀取一行數(shù)據(jù) fgets()/fgetcsv()

4.一次讀完全部數(shù)據(jù) fpassthru()/ file()

1. 一次讀取一個(gè)字節(jié) —— 通過fgetc()獲取單個(gè)字節(jié)

<?php
   $DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];
   $fp = fopen("$DOCUMENT_ROOT/text.txt",'r');//打開文件
   if(file_exists("$DOCUMENT_ROOT/text.txt")){//當(dāng)文件存在時(shí),才讀取內(nèi)容
     while(!feof($fp)){//判斷文件指針是否到達(dá)末尾
        $c = fgetc($fp);//每執(zhí)行一次fgetc(),文件指針就向后移動(dòng)一位
        echo $c;//輸出獲取到的字節(jié)
      }
    }
   fclose($fp);//關(guān)閉文件
?>

運(yùn)行:

PHP如何實(shí)現(xiàn)文件寫入和讀取

【注意】:無論是按文本格式輸入輸出還是按二進(jìn)制格式輸出,fgetc()每次獲取的是一個(gè)字節(jié)而不是一個(gè)字符!

上面的例子中我們是逐個(gè)輸出,現(xiàn)在讓我們只做一次輸出,看看結(jié)果怎樣:

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

運(yùn)行結(jié)果如下,我們得到的不是漢字“我”,而是一個(gè)亂碼,其實(shí)這個(gè)亂碼就是一個(gè)字節(jié)

PHP如何實(shí)現(xiàn)文件寫入和讀取

<?php
   $DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];
   $fp = fopen("$DOCUMENT_ROOT/text.txt",'r');
   echo fgetc($fp);//連續(xù)做三次輸出
   echo fgetc($fp);
   echo fgetc($fp);
   fclose($fp);
?>

2.一次讀取多個(gè)字節(jié) ——通過fread()方法:

<?php
  $DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];
  $fp = fopen("$DOCUMENT_ROOT/text.txt",'r');
  echo fread($fp, 3);//一次輸出三個(gè)字節(jié)即一個(gè)漢字字符(UTF-8)
  fclose($fp);
?>

運(yùn)行結(jié)果:

PHP如何實(shí)現(xiàn)文件寫入和讀取

改成:

echo fread($fp, 6);

運(yùn)行結(jié)果如下,輸出了6個(gè)字節(jié)也即兩個(gè)漢字字符(UTF-8)

PHP如何實(shí)現(xiàn)文件寫入和讀取

3.一次讀取一行——通過fgets()獲取一行內(nèi)容

<?php
    $DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT']
    $fp = fopen("$DOCUMENT_ROOT/text.txt",'r');//打開文件
    if(file_exists("$DOCUMENT_ROOT/text.txt")){//當(dāng)文件存在時(shí),才讀取內(nèi)容
     while(!feof($fp)){//判斷文件指針是否到達(dá)末尾
       $line = fgets($fp);//返回一行文本,并將文件指針移動(dòng)到下一行頭部
       echo $line."<br/>";//輸出獲取到的一行文本
     }
    }
    fclose($fp);//關(guān)閉文件
?>

PHP如何實(shí)現(xiàn)文件寫入和讀取

fgets()其實(shí)還有第二個(gè)參數(shù),這個(gè)參數(shù)規(guī)定了每一行能讀取的最大字節(jié)數(shù)(注意是字節(jié)數(shù)不是字符數(shù)):

【注意】在UTF-8編碼下漢字3字節(jié),字母1字節(jié)

下面我修改上面的一行,代碼,使獲取的每一行最大字符數(shù)為3(也即字節(jié)數(shù)為9)

$line = fgets($fp,10);

Demo:

PHP如何實(shí)現(xiàn)文件寫入和讀取

【注意】:這里我fgets()里第二個(gè)參數(shù)為10,為什么是10呢?因?yàn)?/p>

1.這里的長度是按字節(jié)數(shù)算的

2.一個(gè)漢字占3個(gè)字節(jié)。fgets($fp,10)代表一次最多讀取10 - 1 = 9字節(jié)

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

fpassthru()將讀取文件并直接輸出(無處理過程)

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

運(yùn)行結(jié)果:

PHP如何實(shí)現(xiàn)文件寫入和讀取

【注意】這里需要注意一點(diǎn)的是,我們并沒有從fpassthru($fp)獲取到返回值然后echo到頁面上去,也就是說這個(gè)方法是會(huì)強(qiáng)制輸出獲取的內(nèi)容的,而并不是像之前例子的方法那樣返回文本,允許我們保存到變量中才將其輸出

將讀取到的全部內(nèi)容保存到一個(gè)數(shù)組中,每個(gè)數(shù)組元素為一行的內(nèi)容——fille()

<?php
  $DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];
  $file_array = file("$DOCUMENT_ROOT/text.txt");//取到文件數(shù)組
  foreach ($file_array as $value) {//輸出數(shù)組元素
    echo $value."<br/>";
  }
?>

PHP如何實(shí)現(xiàn)文件寫入和讀取

【注意】:這里我們并不需要寫fopen和fclose哦!也就是說file()方法已經(jīng)幫我們做了這一步了

四.使用fclose方法關(guān)閉文件

fclose()將返回一個(gè)布爾值,成功關(guān)閉為true,關(guān)閉失敗為false(失敗的情況很少出現(xiàn),可不考慮)

是否打開文件后一定要關(guān)閉?

1即使不手寫fclose,在PHP腳本執(zhí)行結(jié)束后,也會(huì)自動(dòng)關(guān)閉文件的

2但在一個(gè)長時(shí)間執(zhí)行的腳本中,如果不寫關(guān)閉文件的fclose(),在文件加鎖的情況下會(huì)造成操作的阻塞,所以,寫fclose是個(gè)好習(xí)慣

五.文件指針的移動(dòng)

我們上面調(diào)用的讀取文件的函數(shù),其實(shí)都是基于文件指針去打印的,每讀取一段字節(jié)內(nèi)容,文件指針就向后移動(dòng)一段字節(jié)長度,直到被讀取的文件最大字節(jié)長度為止

<?php
     $DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];
     function print_file_pointer($fp){//定義一個(gè)打印文件指針位置的函數(shù)
       echo " <br/>//此時(shí)文件指針的位置:";
       echo ftell($fp)."<br/>";
     }
     $fp = fopen("$DOCUMENT_ROOT/text.txt",'r');
     echo fgetc($fp);//通過fgetc連續(xù)輸出三個(gè)字節(jié)
     echo fgetc($fp);
     echo fgetc($fp);
     print_file_pointer($fp);//打印此刻文件指針的位置
     
     echo fread($fp,6);//通過fread一次輸出6字節(jié)
     print_file_pointer($fp);//打印此刻文件指針的位置
     
     echo fgets($fp); //通過fgets輸出一整行
     print_file_pointer($fp);//打印此刻文件指針的位置
     
     fpassthru($fp); //一次性輸出全部內(nèi)容
     print_file_pointer($fp);//打印此刻文件指針的位置
     
     fseek($fp, 33);//使文件指針移動(dòng)到33字節(jié)位置
     print_file_pointer($fp);//打印此刻文件指針的位置
     
     rewind($fp);//使文件指針移動(dòng)到0字節(jié)位置(初始位置)
     print_file_pointer($fp);//打印此刻文件指針的位置
$fclose($fp);
?>

Demo:

PHP如何實(shí)現(xiàn)文件寫入和讀取

所以我們需要正確理解fgets(),fpassthru()這些函數(shù)的作用:

fgets():從當(dāng)前文件指針的位置到本行結(jié)束的數(shù)據(jù),而不是一定輸出一整行

fpassthru():從當(dāng)前文件指針的位置到全部內(nèi)容結(jié)束的數(shù)據(jù),而不是一定輸出所有的數(shù)據(jù)

但在這里你可能會(huì)有疑問:為什么輸出“湖灣”后的指針位置會(huì)是17而不是15呢?按理說輸出“我叫彭湖灣”這5個(gè)漢字一共占3*5 = 15個(gè)字節(jié),多出來的17 - 15 =2字節(jié)是什么呢?

多出來的兩個(gè)字節(jié)是windows下的回車換行符\n\r

\n是換行,占一字節(jié),\r是回車,占一字節(jié),在六中我將會(huì)介紹

六.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);
?>

我們?cè)趙indows下敲下回車鍵的時(shí)候,相當(dāng)于鍵入了\n\r,所以“我叫彭湖灣”的15字節(jié)+“\n\r”的2字節(jié) = 17字節(jié)

PHP如何實(shí)現(xiàn)文件寫入和讀取

在mac下不一樣的是:敲下回車鍵的時(shí)候,相當(dāng)于只鍵入了\n,所以“我叫彭湖灣”的15字節(jié)+“\n”的1字節(jié) = 16字節(jié)

PHP如何實(shí)現(xiàn)文件寫入和讀取

到此,相信大家對(duì)“PHP如何實(shí)現(xiàn)文件寫入和讀取”有了更深的了解,不妨來實(shí)際操作一番吧!這里是億速云網(wǎng)站,更多相關(guān)內(nèi)容可以進(jìn)入相關(guān)頻道進(jìn)行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI