fopen 方法可能是以前的 C 和 C++ 程式員最熟悉的,因為如果您使用過這些語言,那麼它們或多或少都是您已掌握多年的工具。對於這些方法中的任何一種,通過使用 fopen(用於讀取資料的函數)的標準方法開啟檔案,然後使用 fclose 關閉檔案,如清單 1 所示。
清單 1. 用 fgets 開啟並讀取檔案
$file_handle = fopen("myfile", "r"); while (!feof($file_handle)) { $line = fgets($file_handle); echo $line; } fclose($file_handle);
開啟檔案。$file_handle 儲存了一個對檔案本身的引用。
檢查您是否已到達檔案的末尾。
繼續讀取檔案,直至到達檔案末尾,邊讀取邊列印每行。
關閉檔案。
記住這些步驟。
feof
feof 命令將檢測您是否已經讀到檔案的末尾並返回 True 或 False。清單 1 中的迴圈將繼續執行,直至您達到檔案“myfile”的末尾。註:如果讀取的是 URL 並且通訊端由於不再有任何資料可以讀取而逾時,則 feof 也將返回 False。
fclose
向前跳至清單 1 的末尾,fclose 將實現與 fopen 相反的功能:它將關閉指向檔案或 URL 的串連。執行此函數後,您將不再能夠從檔案或通訊端中讀取任何資訊。
fgets
在清單 1 中回跳幾行,您就到達了檔案處理的核心:實際讀取檔案。fgets 函數是處理第一個樣本的首選武器。它將從檔案中提取一行資料並將其作為字串返回。在那之後,您可以列印或者以別的方式處理資料。清單 1 中的樣本將精細地列印整個檔案。
如果決定限制處理資料區塊的大小,您可以將一個參數添加到 fgets 中限制最大行長度。例如,使用以下代碼將行長度限制為 80 個字元:$string = fgets($file_handle, 81);
回想 C 中的“\0”字串末尾終止符,將長度設為比實際所需值大一的數字。因而,如果需要 80 個字元,則以上樣本使用 81。應養成以下習慣:只要對此函數使用行限制,就添加該額外字元。
fread
fgets 函數是多個檔案讀取函數中惟一一個可用的。它是一個更常用的函數,因為逐行解析通常會有意義。事實上,幾個其他函數也可以提供類似功能。但是,您並非總是需要逐行解析。
這時就需要使用 fread。fread 函數與 fgets 的處理目標略有不同:它趨於從二進位檔案(即,並非主要包含人類可閱讀的文本的檔案)中讀取資訊。由於“行”的概念與二進位檔案無關(邏輯資料結構通常都不是由新行終止),因此您必須指定需要讀入的位元組數。$fh = fopen("myfile", "rb");
$data = fread($file_handle, 4096);
<?php$file_path = "test.txt";if(file_exists($file_path)){ $fp = fopen($file_path,"r"); $str = fread($fp,filesize($file_path));//指定讀取大小,這裡把整個檔案內容讀取出來 echo $str = str_replace("\r\n","<br />",$str);}?>
使用位元據
注意:此函數的樣本已經使用了略微不同於 fopen 的參數。當處理位元據時,始終要記得將 b 選項包含在 fopen 中。如果跳過這一點,Microsoft Windows 系統可能無法正確處理檔案,因為它們將以不同的方式處理新行。如果處理的是 Linux 系統(或其他某個 UNIX 變種),則這可能看似沒什麼關係。但即使不是針對 Windows 開發的,這樣做也將獲得良好的跨平台可維護性,並且也是應當遵循的一個好習慣。
以上代碼將讀取 4,096 位元組 (4 KB) 的資料。註:不管指定多少位元組,fread 都不會讀取超過 8,192 個位元組 (8 KB)。
假定檔案大小不超過 8 KB,則以下代碼應當能將整個檔案讀入一個字串。$fh = fopen("myfile", "rb");
$data = fread($fh, filesize("myfile"));
fclose($fh);
如果檔案長度大於此值,則只能使用迴圈將其餘內容讀入。
fscanf
回到字串處理,fscanf 同樣遵循傳統的 C 檔案庫函數。如果您不熟悉它,則 fscanf 將把欄位資料從檔案讀入變數中。list ($field1, $field2, $field3) = fscanf($fh, "%s %s %s");
此函數使用的格式字串在很多地方都有描述(如 PHP.net 中),故在此不再贅述。可以這樣說,字串格式化極為靈活。值得注意的是所有欄位都放在函數的傳回值中。(在 C 中,它們都被作為參數傳遞。)
fgetss
fgetss 函數不同於傳統檔案函數並使您能更好地瞭解 PHP 的力量。該函數的功能類似於 fgets 函數,但將去掉發現的任何 HTML 或 PHP 標記,只留下純文字。查看如下所示的 HTML 檔案。
範例 HTML 檔案
<html> <head><title>My title</title></head> <body> <p>If you understand what "Cause there ain't no one for to give you no pain" means then you listen to too much of the band America</p> </body> </html>
然後通過 fgetss 函數過濾它。
使用 fgetss
$file_handle = fopen("myfile", "r"); while (!feof($file_handle)) { echo = fgetss($file_handle); } fclose($file_handle);
以下是輸出: My title
If you understand what "Cause there ain't no one for to give you no pain"
means then you listen to too much of the band America
fpassthru 函數
無論怎樣讀取檔案,您都可以使用 fpassthru 將其餘資料轉儲到標準輸出通道。fpassthru($fh);
此外,此函數將列印資料,因此無需使用變數擷取資料。
非線性檔案處理:跳躍訪問
當然,以上函數只允許順序讀取檔案。更複雜的檔案可能要求您來回跳轉到檔案的不同部分。這時就用得著 fseek 了。fseek($fh, 0);
以上樣本將跳回檔案的開頭。如果不需要完全返回 —— 我們可設定返回KB —— 然後就可以這樣寫:fseek($fh, 1024);
從 PHP V4.0 開始,您有一些其他選項。例如,如果需要從當前位置向前跳轉 100 個位元組,則可以嘗試使用:fseek($fh, 100, SEEK_CUR);
類似地,可以使用以下代碼向後跳轉 100 個位元組:fseek($fh, -100, SEEK_CUR);
如果需要向後跳轉至檔案末尾前 100 個位元組處,則應使用 SEEK_END。fseek($fh, -100, SEEK_END);
在到達新位置後,可以使用 fgets、fscanf 或任何其他方法讀取資料。
註:不能將 fseek 用於引用 URL 的檔案處理。
回頁首
提取整個檔案
現在,我們將接觸到一些 PHP 的更獨特的檔案處理功能:用一兩行處理大塊資料。例如,如何提取檔案並在 Web 頁面上顯示其全部內容?好的,您看到了 fgets 使用迴圈的樣本。但是如何能夠使此過程變得更簡單?用 fgetcontents 會使過程超級簡單,該方法將把整個檔案放入一個字串中。$my_file = file_get_contents("myfilename");
echo $my_file;
雖然它不是最好的做法,但是可以將此命令更簡明地寫為:echo file_get_contents("myfilename");
本文主要介紹的是如何處理本地檔案,但是值得注意的是您還可以用這些函數提取、回顯和解析其他 Web 頁面。echo file_get_contents("http://127.0.0.1/");
此命令等效於:$fh = fopen("http://127.0.0.1/", "r");
fpassthru($fh);
您一定會查看此命令並認為:“那還是太費力”。PHP 開發人員同意您的看法。因此可以將以上命令縮短為:readfile("http://127.0.0.1/");
readfile 函數將把檔案或 Web 頁面的全部內容轉儲到預設的輸出緩衝區。預設情況下,如果失敗,此命令將列印錯誤訊息。要避免此行為(如果需要),請嘗試:@readfile("http://127.0.0.1/");
當然,如果確實需要解析檔案,則 file_get_contents 返回的單個字串可能有些讓人吃不消。您的第一反應可能是用 split() 函數將它分解一下。$array = split("\n", file_get_contents("myfile"));
但是既然已經有一個很好的函數為您執行此操作為什麼還要這樣大費周章?PHP 的 file() 函數一步即可完成此操作:它將返回分為若干行的字串數組。$array = file("myfile");
應當注意的是,以上兩個樣本有一點細微差別。雖然 split 命令將刪除新行,但是當使用 file 命令(與 fgets 命令一樣)時,新行仍將被附加到數組中的字串上。
但是,PHP 的力量還遠不止於此。您可以在一條命令中使用 parse_ini_file 解析整個 PHP 樣式的 .ini 檔案。
當然,您可能注意到此命令合并了各個部分。這是預設行為,但是您可以通過將第二個參數傳遞給 parse_ini_file 輕鬆地修正它:process_sections,這是一個布爾型變數。將 process_sections 設為 True。
$file_array = parse_ini_file("holy_grail.ini", true); print_r $file_array;
並且您將獲得以下輸出:
Array ( [personal information] => Array ( [name] => King Arthur [quest] => To seek the Holy Grail [favorite color] => Blue ) [more stuff] => Array ( [Samuel Clemens] => Mark Twain [Caryn Johnson] => Whoopi Goldberg ) )
<?php$file_path = "test.txt";if(file_exists($file_path)){ $fp = fopen($file_path,"r"); $str = ""; $buffer = 1024;//每次讀取 1024 位元組 while(!feof($fp)){//迴圈讀取,直至讀取完整個檔案 $str .= fread($fp,$buffer); } $str = str_replace("\r\n","<br />",$str); echo $str;}?>