【PHP】文件寫入和讀取詳解

来源:http://www.cnblogs.com/penghuwan/archive/2017/05/21/6884932.html
-Advertisement-
Play Games

最近在學習PHP的文件讀寫時,感覺很多資料對文件打開模式說得不很清楚,特別是關於r+,w+,a+三者的區別,對文件指針的概念也說的很模糊。所以我就將自己學習歷程系統寫了下來。本文包括:一.實現文件讀取和寫入的基本思路 二.使用fopen方法打開文件 三.文件讀取和文件寫入操作 四.使用fclose方... ...


文章提綱: 一.實現文件讀取和寫入的基本思路 二.使用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位元組     【完】
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 流迭代器 2017-05-21 17:05:51 流迭代器是標準模板庫STL中的,是類模板,流迭代器實例化之後即可以和任何接受對應迭代器的函數一起使用(可以將流看做一個容器,把數據存儲在一個連續的緩衝區中,具有迭代器的功能和類似使用)。 istream_iterator 和ostream_itera ...
  • 因為原文中延續了組合模式的代碼示例來講訪問者模式 所以這裡就合併一起來複習了。但主要還是講訪問者模式。顧名思義這個模式會有一個訪問者類(就像近期的熱播劇“人民的名義”中的檢查官,跑到到貪官家裡調查取證,查實後就定罪),被訪問者類調用訪問者類的時候會將自身傳遞給它使用。直接看代碼: //被訪問者基類 ...
  • 1.順序查找 在查找中我們一個一個順序的遍歷表中的所有鍵並使用equals()方法來查找匹配的鍵。 優點:對數組的結構沒有特定的要求,可以使用數組或者鏈表實現,演算法簡單。 缺點:當數組個數n較大時,效率低下。 時間複雜度:查找命中時,最大時間複雜度是O(n),最小時間複雜度是O(1),平均時間複雜度 ...
  • There is 4 method of the referenced method: Student class: StudentComparator class: Test:public class MethodReferenceTest { ...
  • The most crucial distinguish between functional interface invoking and traditional method invoking is that transforming behavious or params it is.Howe ...
  • PHP是什麼文件? PHP,一個嵌套的縮寫名稱,是英文“超級文本預處理語言”(PHP:Hypertext Preprocessor)的縮寫。PHP 是一種 HTML 內嵌式的語言,PHP與微軟的ASP頗有幾分相似,都是一種在伺服器端執行的“嵌入HTML文檔的腳本語言”,語言的風格有類似於C語言,現在 ...
  • 轉載地址:http://www.yihaomen.com/article/java/302.htm mybatis中文官方教程:http://www.mybatis.org/mybatis-3/zh/getting-started.html 觀後提煉小結: 1、介面中的方法名 selectUserB ...
  • CopyOnWriteArrayList是Java併發包中提供的一個併發容器,它是個線程安全且讀操作無鎖的ArrayList,寫操作則通過創建底層數組的新副本來實現,是一種讀寫分離的併發策略,我們也可以稱這種容器為"寫時複製器",Java併發包中類似的容器還有CopyOnWriteSet。本文會對C ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...