【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是追加(不刪除原有內容),這都好理解。php

但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」windows

基於」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字節
 
 
【完】
相關文章
相關標籤/搜索