我們知道,臨時(shí)聲明的變量是保存在內(nèi)存中的,即便是靜態(tài)變量,在腳本運(yùn)行完畢后也會(huì)被釋放掉,so,想長久保存一個(gè)變量的內(nèi)容,方法之一就是寫到文件中,放到硬盤或服務(wù)器上,為此文件操作就必須很熟悉。
1.文件的屬性信息獲取
首先文件具有類型,在Linux下邊,有block(塊設(shè)備,如磁盤分區(qū)、CD-ROM)、char(以字符為輸入的設(shè)備,如鍵盤、打印機(jī))、dir(目錄類型,目錄也是文件的一種)、fifo(命名管道,解釋是將信息從一個(gè)進(jìn)程傳到另一個(gè)進(jìn)程)、file(普通的文件)、link(鏈接,類似win下邊的快捷方式)、unknow(未知類型)7大類,在win下邊,只有3類:file、dir和unknown。Linux渣表示一定要好好搞一下Linux-_-,人家完全是為Linux而生。
關(guān)于類型的獲取有這么幾個(gè)函數(shù):filetype:獲取類型; is_file:判斷為是否是正常文件; is_link:判斷是否是鏈接。
關(guān)于屬性的獲取有這么幾個(gè)函數(shù):
file_exists:判斷文件或目錄是否存在;
filesize:獲取文件大?。?/p>
is_readable、is_writable、is_executable :是否可讀、可寫、可執(zhí)行;
filectime、filemtime、fileatime:獲取文件的創(chuàng)建時(shí)間(create)、修改時(shí)間(modify)、訪問時(shí)間(access),均返回時(shí)間戳;
stat:獲取文件的一些基本信息,返回一個(gè)索引與關(guān)聯(lián)混合數(shù)組。
比如,可以這樣判斷文件類型:
<?php function getFileType($path){ // 獲取文件類型 switch(filetype($path)){ case 'file': return 'ordinary file'; case 'dir': return 'directory'; case 'block': return 'block device file'; case 'char': return 'transfer device base on char'; case 'fifo': return 'named pipes'; case 'link': return 'symbol link'; default: return 'unknown type'; } }
filesize返回的是以字節(jié)為單位的數(shù)據(jù),如果是大文件數(shù)字或很大,可以對(duì)數(shù)字先處理一下,代碼如下
<?php // 處理文件大小 function getSize($path = '', $size = -1){ if($path !== null && $size == -1){ // 只傳路徑就計(jì)算大小,也可以使之只處理數(shù)字 $size = filesize($path); } if($size >= pow(2, 40)){ return round($size/pow(2, 40), 2).'TB'; } else if($size >= pow(2, 30)){ return round($size/pow(2, 30), 2).'GB'; } else if($size >= pow(2, 20)){ return round($size/pow(2, 20), 2).'MB'; } else if($size >= pow(2, 10)){ return round($size/pow(2, 10), 2).'KB'; } else{ return round($size, 2).'Byte'; } }
現(xiàn)在綜合來獲取一下文件信息,代碼如下:
<?php function getFileInfo($path){ if(!file_exists($path)){ // 判斷文件是否存在 echo 'file not exists!<br>'; return; } if(is_file($path)){ // 是文件,打印基礎(chǔ)文件名 echo basename($path).' is a file<br>'; } if(is_dir($path)){ // 是目錄 ,返回目錄 echo dirname($path).' is a directory<br>'; } echo 'file type:'.getFileType($path).'<br>'; // 獲取文件類型 echo 'file size:'.getSize($path).'<br>'; // 獲取文件大小 if(is_readable($path)){ // 是否可讀 echo basename($path).' is readable<br>'; } if(is_writeable($path)){ // 是否可寫 echo basename($path).' is writeable<br>'; } if(is_executable($path)){ // 是否可執(zhí)行 echo basename($path).' is executable<br>'; } // touch函數(shù)可以修改這些時(shí)間 echo 'file create time: '.date('Y-m-d H:i:s', filectime($path)).'<br>'; // 創(chuàng)建時(shí)間 echo 'file modify time: '.date('Y-m-d H:i:s', filemtime($path)).'<br>'; // 修改時(shí)間 echo 'last access time: '.date('Y-m-d H:i:s', fileatime($path)).'<br>'; // 上次訪問時(shí)間 echo 'file owner: '.fileowner($path).'<br>'; // 文件擁有者 echo 'file permission: '.substr(sprintf('%o', (fileperms($path))), -4).'<br>'; // 文件權(quán)限,八進(jìn)制輸出 echo 'file group: '.filegroup($path).'<br>'; // 文件所在的組 }
效果如下:
代碼中還用到了文件權(quán)限、所在組等函數(shù),有必要解釋下(說的不對(duì)請(qǐng)修正)。一個(gè)文件的權(quán)限分為可讀可寫可執(zhí)行,一般這樣表示:rwx,字母對(duì)應(yīng)的表示可讀可寫可執(zhí)行,從前往后規(guī)定值為4、2、1,三個(gè)值相加的結(jié)果最大為7,因此0666用的是八進(jìn)制表示,這樣看起來就很方便。為7則表示這個(gè)文件具備這三個(gè)權(quán)限,那為什么打印的是0666呢?我們都知道,進(jìn)入windows下面是有一個(gè)用戶的,在Linux下邊,與windows類似,也是有一個(gè)用戶登錄進(jìn)去,因此一個(gè)文件可能為該用戶所有,一個(gè)用戶它還有自己所在的組,以及該系統(tǒng)中還有其他組(猜想這樣分應(yīng)該是管理上的需要),因此對(duì)于0666,對(duì)于第一個(gè)6,表示該用戶對(duì)該文件的權(quán)限,第二個(gè)6表示該用戶所在的組對(duì)該文件的權(quán)限,第三個(gè)6表示其他的組所具有的權(quán)限(這樣就不用一一去區(qū)分除本組外其他的用戶了),6就知道該文件是可讀可寫的(win下可執(zhí)行都知道是.exe文件)。
2.目錄操作
目錄的讀取,opendir:打開一個(gè)目錄,返回一個(gè)句柄,指向該目錄中的內(nèi)容,如果把目錄中的內(nèi)容看成一個(gè)有順序的數(shù)據(jù),比如按順序的排列的數(shù)組,這個(gè)句柄就指向這個(gè)數(shù)組的開頭,事實(shí)上,系統(tǒng)會(huì)把該目錄中的內(nèi)容按照字典排序,無論是文件還是子目錄。readdir:讀取下一個(gè)目錄內(nèi)容,返回文件名,并自動(dòng)指向該目錄中的下一個(gè)文件/目錄,所以讀取一個(gè)目錄中的內(nèi)容,不包括子目錄中的內(nèi)容,需要一個(gè)循環(huán)來控制,在讀取完后,還要關(guān)閉句柄變量,C語言讀取文件時(shí)也是這樣,打開就有關(guān)閉。以我的機(jī)子舉例:
<?php // 目錄的讀取 $dir = 'F:/'; echo 'details in '.$dir.'<br>'; if(is_dir($dir)){ if(($handle = opendir($dir)) == false){ // 獲取目錄句柄 echo 'open dir failed'; return; } while(($name = readdir($handle)) != false){ // 循環(huán)讀取該目錄下內(nèi)容 $filepath = $dir.'/'.$name; echo 'name: '.$name.' type: '.filetype($filepath).'<br>'; } closedir($handle); // 關(guān)閉目錄句柄 } else{ echo $dir.' is not a directory<r>'; }
效果如下:
可以看到實(shí)際上,系統(tǒng)給目錄中內(nèi)容進(jìn)行了忽略大小寫的字典排序。
目錄的大小計(jì)算,我們知道文件的大小可以由filesize取得,但是php中沒有專門計(jì)算目錄大小的函數(shù)。當(dāng)然php中有計(jì)算硬盤大小的函數(shù)disk_total_space(計(jì)算硬盤總空間)、disk_free_space(計(jì)算硬盤可用空間),但是我試了下disk_free_space,貌似計(jì)算得不對(duì)。正因?yàn)橛衒ilesize計(jì)算文件的大小,因此,需要用到遞歸,當(dāng)是目錄時(shí),進(jìn)去繼續(xù)計(jì)算子目錄的大小,如果是文件,獲取到文件大小并加上返回,代碼如下:
<?php // 目錄大小計(jì)算 function getDirSize($dirpath){ $size = 0; if(false != ($handle = opendir($dirpath))){ while(false != ($file = readdir($handle))){ if($file == '.' || $file == '..') //注意過濾目錄中自帶的點(diǎn)和點(diǎn)點(diǎn) continue; $filepath = $dirpath.'/'.$file; // 前面要接上路徑 if(is_file($filepath)){ // 是文件計(jì)算大小 $size += filesize($filepath); } else if(is_dir($filepath)){ // 是目錄繼續(xù)計(jì)算該目錄下的文件 $size += getDirSize($filepath); } else{ $size += 0; } } closedir($handle); } return $size; } $dirsize = 'F:/size'; $size = getDirSize($dirsize); echo 'dir size: '.getSize(null, $size).'<br><br>'; // 調(diào)用前面的數(shù)據(jù)處理函數(shù)
我在F盤建了個(gè)size文件,隨便弄了點(diǎn)子目錄和文檔,效果如下,左邊是程序求得,右邊是右鍵查看文件夾屬性得到的,用以對(duì)比。
目錄的新建和刪除,主要用到,mkdir:新建一個(gè)目錄,rmdir:刪除一個(gè)非空目錄,注意只能是非空,代碼如下:
<?php // 目錄的新建和刪除 $newDirPath = 'F:/newDir'; if(true == @mkdir($newDirPath, 0777, true)){ // 加@是因?yàn)槲募汛嬖跁r(shí)php本身可能會(huì)拋出一個(gè)warning echo 'create directory '.$newDirPath.' successfully<br>'; } else{ if(file_exists($newDirPath)) echo 'directory '.$newDirPath.' has existed<br>'; else echo 'create directory '.$newDirPath.' failed<br>'; } if(true == @rmdir('F:/aaa')) //只能刪除非空目錄,如果刪除不存在的目錄自動(dòng)拋出warning echo 'remove successfully<br>';
那么問題來了,如果要?jiǎng)h除一個(gè)非空目錄咋辦,又得自己寫了,思想仍然是遞歸,因?yàn)閜hp只提供了刪除文件函數(shù)unlink,所以在刪除一個(gè)目錄時(shí),先opendir,再進(jìn)入,如果是文件直接刪除,如果是目錄,繼續(xù)進(jìn)入使用該方法處理,當(dāng)然還可已返回一個(gè)bool變量表示刪除是否成功,代碼如下:
<?php // 刪除文件 unlink // 刪除目錄中的內(nèi)容,然后刪除該目錄 function clearDir($dirpath){ if(file_exists($dirpath)){ if(false != ($handle = opendir($dirpath))){ while(false != ($name = readdir($handle))){ if($name == '.' || $name == '..') continue; $filename = $dirpath.'/'.$name; if(is_dir($filename)) clearDir($filename); if(is_file($filename)) @unlink($filename); } closedir($handle); rmdir($dirpath); } else{ return false; } } else{ return false; } return true; }
在這里不得不說遇到的一個(gè)大坑,就是 . 和 .. 這兩個(gè)鬼玩意兒(點(diǎn)和點(diǎn)點(diǎn)),在操作系統(tǒng)中的每一個(gè)文件夾下邊,都會(huì)有 . 和 .. ,它們表示當(dāng)前目錄和當(dāng)前目錄的上級(jí)目錄,可惡的是前面在讀取目錄時(shí)居然沒顯示,導(dǎo)致遞歸函數(shù)成了死循環(huán),因?yàn)?. 和 .. 在每一個(gè)目錄的最前面,必然會(huì)先讀到它倆,若不過濾,首先讀到 . ,它表示本目錄,然后又遞歸進(jìn)入本目錄...這倆是操作系統(tǒng)下面的默認(rèn)有的,它們是本目錄與上級(jí)目錄的連接符。
通過計(jì)算目錄的大小和刪除非空目錄的代碼,寫復(fù)制和剪切目錄就非常容易,非常相似的遞歸思想,需要用到復(fù)制文件函數(shù)copy,文件移動(dòng)函數(shù)rename,這個(gè)挺有趣,rename,字面上是重命名,但是重命名到另一個(gè)目錄中不就是剪切了么-_-
3.文件讀寫
php的某些文件讀取操作跟C語言非常像,所以也比較簡單,步驟就是先打開文件獲取句柄,檢查錯(cuò)誤,然后讀寫處理,然后關(guān)閉,養(yǎng)成打開處理完后就關(guān)閉的好習(xí)慣,記得在C語言中的文件不關(guān)閉的話,打開兩次是會(huì)報(bào)錯(cuò)滴,不知道記錯(cuò)沒,所以嚴(yán)格點(diǎn)的程序都有非常多的處理,比如先驗(yàn)證文件存在,然后驗(yàn)證可讀可寫性,然后先關(guān)閉一下,然后再打開,打開時(shí)還得再檢查打開錯(cuò)了沒......在打開文件時(shí),就要選擇打開文件的模式,它決定了我們讀還是寫文件,當(dāng)然是對(duì)需要這樣操作的函數(shù)有用。
寫文件,寫文件函數(shù)只有fwrite、fputs、file_put_contents少數(shù)幾個(gè),其中fwrite與fputs效果一樣,file_put_contents是一次性向文件寫入一些內(nèi)容,它就不需要指定打開模式,同時(shí)它也可以是附加或者覆蓋現(xiàn)有文件內(nèi)容,比如:
<?php // 寫 fwrite(別名fputs) $filepath = 'F:/10m.txt'; function writeSome($filepath){ if(($handle = fopen($filepath, 'r+')) == true){ for($i=0; $i<10; $i++) fwrite($handle, $i." write something/r/n"); // windws以/r/n作為換行符 fclose($handle); } } file_put_contents($filepath, 'use file_put_contents function', FILE_APPEND); // 附加內(nèi)容
讀文件,讀文件的函數(shù)多些,有fread(讀取指定個(gè)字節(jié))、fgetc(讀取一個(gè))、fgets(讀取一行)、file(全部讀取,按行分配到一個(gè)數(shù)組中返回)、file_get_contents(默認(rèn)讀取全部返回字符串)、readfile(直接將文件中內(nèi)容輸出到緩存,效果就是直接在瀏覽器上輸出),伴隨著fread、fget、fgets運(yùn)行,文件指針會(huì)自動(dòng)往后走。因此連續(xù)讀最好是循環(huán)控制。讀到文件末尾怎么辦,EOF標(biāo)識(shí)指示到達(dá)文件末尾,最好用feof檢測是否到文件末尾。不多說,看代碼:
<?php // fread讀取 function readSome($filepath){ if(($handle = @fopen($filepath, 'r')) == true){ while(!feof($handle)){ // 判斷是否到達(dá)文件末尾 $str = fread($handle, 10); // fread讀取時(shí),文件指針自動(dòng)向后移動(dòng) echo $str.'<br>'; } } }
如果想要讀取方式更靈活,就要配合fseek、rewind使用,它們可以移動(dòng)文件指針到具體位置,fseek十分靈活,可以直接移到開頭或末尾,或從當(dāng)前位置往前或后移動(dòng),讀取想要的內(nèi)容,ftell還可告知當(dāng)前位置,比如:
<?php function readFun($filepath){ if(($handle = @fopen($filepath, 'r')) != false){ echo 'html' target='_blank'>current position: '.ftell($handle).'<br>'; // 輸出文件當(dāng)前文件指針位置,以字節(jié)算,0表示開頭 $str = fread($handle, 3); // 讀取3個(gè)字節(jié),同時(shí)指針自動(dòng)后移3個(gè)字節(jié) echo 'read content: '.$str.'<br>'; echo 'current position: '.ftell($handle).'<br>'; fseek($handle, 5, SEEK_CUR); // 將文件指針從當(dāng)前位置后移5個(gè)字節(jié) echo 'current position: '.ftell($handle).'<br>'; $str = fread($handle, 5); echo 'read content: '.$str.'<br>'; echo 'current position: '.ftell($handle).'<br>'; rewind($handle); // 返回文件開頭 echo 'current position: '.ftell($handle).'<br>'; fseek($handle, 0, SEEK_END); // 移到文件末尾 echo 'current position: '.ftell($handle).'<br>'; fclose($handle); // 關(guān)閉文件 } }
比如我現(xiàn)在使用該方法讀取一個(gè)寫有從a到z的文本文件,看看效果:
鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請(qǐng)第一時(shí)間聯(lián)系我們修改或刪除,多謝。
新聞熱點(diǎn)
疑難解答
圖片精選