【PHP】文件写入和读取详解
生活随笔
收集整理的這篇文章主要介紹了
【PHP】文件写入和读取详解
小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
文章提綱: 一.實現(xiàn)文件讀取和寫入的基本思路 二.使用fopen方法打開文件 三.文件讀取和文件寫入操作 四.使用fclose方法關閉文件 五.文件指針的移動 六.Windows和UNIX下的回車和換行 ? 一.實現(xiàn)文件讀取和寫入的基本思路: 1.通過fopen方法打開文件:$fp =fopen(/*參數(shù),參數(shù)*/),fp為Resource類型 2.進行文件讀取或者文件寫入操作(這里使用的函數(shù)以1中返回的$fp作為參數(shù)) 3. ? 調用fclose($fp)關閉關閉文件 二:使用fopen方法打開文件 fopen(文件路徑[string],打開模式[string]) <1>fopen的第一個參數(shù)為文件路徑 寫文件路徑的方式:1絕對路徑,2相對路徑 1絕對路徑: 在windows下工作的小伙伴們應該很熟悉,windows下的路徑分隔符是“\”而不是“/”,但我們在寫入路徑時不能以欽定的“\”為分隔符 ?
?
那如果我們以“\”分隔符寫入路徑會怎樣呢? <?php$fp = fopen("C:\wamp64\www\text.txt",'w'); ?> 運行后報錯,提示路徑參數(shù)無效 ??
所以我們要把分隔符“\”換成“/”: <?php$fp = fopen("C:/wamp64/www/text.txt",'w'); ?> 運行時無報錯,說明參數(shù)是有效的。 【注意】fopen函數(shù)不能理解“\”分隔符,如果你想要使用“\”,那么要使用轉義,如寫成:"C:\\wamp64\\www\\text.txt"這種寫法也是可以的,函數(shù)也能理解,不會報錯。但即使這樣,也不推薦使用“\”,因為在OS(mac)下只能識別“/”不能識別“\” 本小節(jié)的結論:推薦堅持使用“/”作為分隔符 2.相對路徑: 上一小節(jié)介紹的是絕對路徑的寫法,但這樣卻帶來了另外一個問題:服務器的目錄結構可能會有較大的改變,這時原來寫的絕對路徑就要全部重寫了,比如在我的電腦上的目標文件路徑是C:/wamp64/www/text.txt,如果我把www文件夾改名為penghuwan呢?原來寫入的路徑參數(shù)就失效了。所以我們引入了相對路徑的寫法: <?php$DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];$fp = fopen("$DOCUMENT_ROOT/text.txt",'w'); ?> ? $_SERVER是PHP的超級全局變量(在代碼任何地方都可訪問,類型是數(shù)組),通過$_SERVER['DOCUMENT_ROOT']可取到服務器的默認根目錄 服務器的默認根目錄可通過php.ini修改(這個可自行百度) ? $_SERVER['DOCUMENT_ROOT']在這里等同于C:/wamp64/www 本小節(jié)的結論:推薦使用相對路徑 <2>fopen的第二個參數(shù)為打開模式 設置打開模式后,我們就相當于為接下來的讀寫操作設置了權限: 最基本的幾個模式: “r”:只能讀取文件,不能寫入文件(寫入操作被忽略) “w”:只能寫入文件,不能讀取文件(讀取操作被忽略) “a”:只追加文件,與“w”類似,區(qū)別是“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,發(fā)現(xiàn)文件內容并沒有改變,說明由于沒有設置相應的權限,操作被忽略了 ? 關于打開模式的網絡資料,我想大家最可能找到的是這張表:(圖來自W3C) ?很全面,但我覺得這張表對新手有些不太友好,讓人看后不知多云。 r是只讀,w是只寫(原來有的內容全刪除),a是追加(不刪除原有內容),這都好理解。
但r+,w+,和a+的區(qū)別和聯(lián)系講的實在太模糊了呀。 這里我就想詳細地講一下r+,w+,和a+三者的區(qū)別和聯(lián)系: 首先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); ?> 運行后再打開文本,發(fā)現(xiàn)“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); ?> 運行后,我們發(fā)現(xiàn)”I am initialized value”已經被刪除了,然后才加上了“w+ mode”這段新文本 ? 【注意】r+,a+,w+還有一個區(qū)別是a+,w+在文件不存在時則創(chuàng)建文件,r+文件不存在時報錯 【吐槽】:關于r+和w+,a+的區(qū)別,我找了網絡上,包括W3C和各種博客文章以及那本“PHP圣經”上的各種資料,發(fā)現(xiàn)都是一筆帶過去的,這也是我寫這篇文章的原因 ? 三.文件讀取和文件寫入操作 先說說幾個比較重要的函數(shù): ? file_exists():判斷文件是否存在,返回布爾值 ? filesize():判斷一個文件大小,返回文件的字節(jié)數(shù),為整型數(shù)字 ? unlink():刪除一個文件 寫入文件 fwrite(資源文件對象[string],寫入方式[string]),資源文件對象即為fopen方法返回的參數(shù),為Resource類型,寫入方式可以是w(或者w+,a+,r+) 已經有上面的例子,這里就不放demo了 讀取文件 這是我們要讀取的文件內容: ??
讀取文件的方式有以下幾種: 1.一次讀取一個字節(jié)的數(shù)據(jù) fgetc() 2.一次讀取指定的字節(jié)數(shù)的數(shù)據(jù) fread() 3.一次讀取一行數(shù)據(jù) fgets()/fgetcsv() 4.一次讀完全部數(shù)據(jù) ?fpassthru()/ file() 1. 一次讀取一個字節(jié) —— 通過fgetc()獲取單個字節(jié) <?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);//每執(zhí)行一次fgetc(),文件指針就向后移動一位echo $c;//輸出獲取到的字節(jié) }}fclose($fp);//關閉文件 ?> 運行: ? 【注意】:無論是按文本格式輸入輸出還是按二進制格式輸出,fgetc()每次獲取的是一個字節(jié)而不是一個字符! 上面的例子中我們是逐個輸出,現(xiàn)在讓我們只做一次輸出,看看結果怎樣: <?php$DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];$fp = fopen("$DOCUMENT_ROOT/text.txt",'r');echo fgetc($fp);//只做一次輸出close($fp); ?> 運行結果如下,我們得到的不是漢字“我”,而是一個亂碼,其實這個亂碼就是一個字節(jié) ? <?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.一次讀取多個字節(jié) ——通過fread()方法: <?php$DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];$fp = fopen("$DOCUMENT_ROOT/text.txt",'r');echo fread($fp, 3);//一次輸出三個字節(jié)即一個漢字字符(UTF-8)fclose($fp); ?>?
運行結果: ?改成:
?
echo fread($fp, 6);?
運行結果如下,輸出了6個字節(jié)也即兩個漢字字符(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()其實還有第二個參數(shù),這個參數(shù)規(guī)定了每一行能讀取的最大字節(jié)數(shù)(注意是字節(jié)數(shù)不是字符數(shù)): 【注意】在UTF-8編碼下漢字3字節(jié),字母1字節(jié) 下面我修改上面的一行,代碼,使獲取的每一行最大字符數(shù)為3(也即字節(jié)數(shù)為9) $line = fgets($fp,10); Demo: ? 【注意】:這里我fgets()里第二個參數(shù)為10,為什么是10呢?因為 1.這里的長度是按字節(jié)數(shù)算的 2.一個漢字占3個字節(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); ?>?
運行結果: ? 【注意】這里需要注意一點的是,我們并沒有從fpassthru($fp)獲取到返回值然后echo到頁面上去,也就是說這個方法是會強制輸出獲取的內容的,而并不是像之前例子的方法那樣返回文本,允許我們保存到變量中才將其輸出 ? 將讀取到的全部內容保存到一個數(shù)組中,每個數(shù)組元素為一行的內容——fille() <?php$DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];$file_array = file("$DOCUMENT_ROOT/text.txt");//取到文件數(shù)組foreach ($file_array as $value) {//輸出數(shù)組元素echo $value."<br/>";} ?>?
??
? 【注意】:這里我們并不需要寫fopen和fclose哦!也就是說file()方法已經幫我們做了這一步了 ? 四.使用fclose方法關閉文件 fclose()將返回一個布爾值,成功關閉為true,關閉失敗為false(失敗的情況很少出現(xiàn),可不考慮) 是否打開文件后一定要關閉? 1即使不手寫fclose,在PHP腳本執(zhí)行結束后,也會自動關閉文件的 2但在一個長時間執(zhí)行的腳本中,如果不寫關閉文件的fclose(),在文件加鎖的情況下會造成操作的阻塞,所以,寫fclose是個好習慣 ? 五.文件指針的移動 我們上面調用的讀取文件的函數(shù),其實都是基于文件指針去打印的,每讀取一段字節(jié)內容,文件指針就向后移動一段字節(jié)長度,直到被讀取的文件最大字節(jié)長度為止 <?php$DOCUMENT_ROOT = $_SERVER['DOCUMENT_ROOT'];function print_file_pointer($fp){//定義一個打印文件指針位置的函數(shù)echo " <br/>//此時文件指針的位置:";echo ftell($fp)."<br/>";}$fp = fopen("$DOCUMENT_ROOT/text.txt",'r');echo fgetc($fp);//通過fgetc連續(xù)輸出三個字節(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); //一次性輸出全部內容print_file_pointer($fp);//打印此刻文件指針的位置fseek($fp, 33);//使文件指針移動到33字節(jié)位置print_file_pointer($fp);//打印此刻文件指針的位置rewind($fp);//使文件指針移動到0字節(jié)位置(初始位置)print_file_pointer($fp);//打印此刻文件指針的位置 $fclose($fp); ?>?
Demo: ? 所以我們需要正確理解fgets(),fpassthru()這些函數(shù)的作用: fgets():從當前文件指針的位置到本行結束的數(shù)據(jù),而不是一定輸出一整行 fpassthru():從當前文件指針的位置到全部內容結束的數(shù)據(jù),而不是一定輸出所有的數(shù)據(jù) 但在這里你可能會有疑問:為什么輸出“湖灣”后的指針位置會是17而不是15呢?按理說輸出“我叫彭湖灣”這5個漢字一共占3*5 ?= 15個字節(jié),多出來的17 - 15 =2字節(jié)是什么呢? 多出來的兩個字節(jié)是windows下的回車換行符\n\r \n是換行,占一字節(jié),\r是回車,占一字節(jié),在六中我將會介紹 ? 六.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字節(jié)+“\n\r”的2字節(jié) = 17字節(jié) ? 在mac下不一樣的是:敲下回車鍵的時候,相當于只鍵入了\n,所以“我叫彭湖灣”的15字節(jié)+“\n”的1字節(jié) = 16字節(jié) 【完】轉載于:https://www.cnblogs.com/penghuwan/p/6884932.html
總結
以上是生活随笔為你收集整理的【PHP】文件写入和读取详解的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Oracle查看IP操作,Oracle
- 下一篇: anaconda利用sns或plt画图中