詳談PHP檔案目錄基礎操作,
我們知道,臨時聲明的變數是儲存在記憶體中的,即便是靜態變數,在指令碼運行完畢後也會被釋放掉,so,想長久儲存一個變數的內容,方法之一就是寫到檔案中,放到硬碟或伺服器上,為此檔案操作就必須很熟悉。
1.檔案的屬性資訊擷取
首先檔案具有類型,在Linux下邊,有block(塊裝置,如磁碟分割、CD-ROM)、char(以字元為輸入的裝置,如鍵盤、印表機)、dir(目錄類型,目錄也是檔案的一種)、fifo(具名管道,解釋是將資訊從一個進程傳到另一個進程)、file(普通的檔案)、link(連結,類似win下邊的捷徑)、unknow(未知類型)7大類,在win下邊,只有3類:file、dir和unknown。Linux渣表示一定要好好搞一下Linux-_-,人家完全是為Linux而生。
關於類型的擷取有這麼幾個函數:filetype:擷取類型; is_file:判斷為是否是正常檔案; is_link:判斷是否是連結。
關於屬性的擷取有這麼幾個函數:
file_exists:判斷檔案或目錄是否存在;
filesize:擷取檔案大小;
is_readable、is_writable、is_executable :是否可讀、可寫、可執行;
filectime、filemtime、fileatime:擷取檔案的建立時間(create)、修改時間(modify)、訪問時間(access),均返回時間戳記;
stat:擷取檔案的一些基本資料,返回一個索引與關聯混合數組。
比如,可以這樣判斷檔案類型:
複製代碼 代碼如下:
<?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返回的是以位元組為單位的資料,如果是大檔案數字或很大,可以對數字先處理一下,代碼如下
複製代碼 代碼如下:
<?php
// 處理檔案大小
function getSize($path = '', $size = -1){
if($path !== null && $size == -1){ // 只傳路徑就計算大小,也可以使之只處理數字
$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';
}
}
現在綜合來擷取一下檔案資訊,代碼如下:
複製代碼 代碼如下:
<?php
function getFileInfo($path){
if(!file_exists($path)){ // 判斷檔案是否存在
echo 'file not exists!
';
return;
}
if(is_file($path)){ // 是檔案,列印基礎檔案名稱
echo basename($path).' is a file
';
}
if(is_dir($path)){ // 是目錄 ,返回目錄
echo dirname($path).' is a directory
';
}
echo 'file type:'.getFileType($path).'
'; // 擷取檔案類型
echo 'file size:'.getSize($path).'
'; // 擷取檔案大小
if(is_readable($path)){ // 是否可讀
echo basename($path).' is readable
';
}
if(is_writeable($path)){ // 是否可寫
echo basename($path).' is writeable
';
}
if(is_executable($path)){ // 是否可執行
echo basename($path).' is executable
';
}
// touch函數可以修改這些時間
echo 'file create time: '.date('Y-m-d H:i:s', filectime($path)).'
'; // 建立時間
echo 'file modify time: '.date('Y-m-d H:i:s', filemtime($path)).'
'; // 修改時間
echo 'last access time: '.date('Y-m-d H:i:s', fileatime($path)).'
'; // 上次訪問時間
echo 'file owner: '.fileowner($path).'
'; // 檔案擁有者
echo 'file permission: '.substr(sprintf('%o', (fileperms($path))), -4).'
'; // 檔案許可權,八進位輸出
echo 'file group: '.filegroup($path).'
'; // 檔案所在的組
}
效果如下:
代碼中還用到了檔案許可權、所在組等函數,有必要解釋下(說的不對請修正)。一個檔案的許可權分為可讀可寫可執行,一般這樣表示:rwx,字母對應的表示可讀可寫可執行,從前往後規定值為4、2、1,三個值相加的結果最大為7,因此0666用的是八進位表示,這樣看起來就很方便。為7則表示這個檔案具備這三個許可權,那為什麼列印的是0666呢?我們都知道,進入windows下面是有一個使用者的,在Linux下邊,與windows類似,也是有一個使用者登入進去,因此一個檔案可能為該使用者所有,一個使用者它還有自己所在的組,以及該系統中還有其他組(猜想這樣分應該是管理上的需要),因此對於0666,對於第一個6,表示該使用者對該檔案的許可權,第二個6表示該使用者所在的組對該檔案的許可權,第三個6表示其他的組所具有的許可權(這樣就不用一一去區分除本組外其他的使用者了),6就知道該檔案是可讀可寫的(win下可執行都知道是.exe檔案)。
2.目錄操作
目錄的讀取,opendir:開啟一個目錄,返回一個控制代碼,指向該目錄中的內容,如果把目錄中的內容看成一個有順序的資料,比如按順序的排列的數組,這個控制代碼就指向這個數組的開頭,事實上,系統會把該目錄中的內容按照字典排序,無論是檔案還是子目錄。readdir:讀取下一個目錄內容,返迴文件名,並自動指向該目錄中的下一個檔案/目錄,所以讀取一個目錄中的內容,不包括子目錄中的內容,需要一個迴圈來控制,在讀取完後,還要關閉控制代碼變數,C語言讀取檔案時也是這樣,開啟就有關閉。以我的機子舉例:
複製代碼 代碼如下:
<?php
// 目錄的讀取
$dir = 'F:/';
echo 'details in '.$dir.'
';
if(is_dir($dir)){
if(($handle = opendir($dir)) == false){ // 擷取目錄控制代碼
echo 'open dir failed';
return;
}
while(($name = readdir($handle)) != false){ // 迴圈讀取該目錄下內容
$filepath = $dir.'/'.$name;
echo 'name: '.$name.' type: '.filetype($filepath).'
';
}
closedir($handle); // 關閉目錄控制代碼
}
else{
echo $dir.' is not a directory';
}
效果如下:
可以看到實際上,系統給目錄中內容進行了忽略大小寫字典排序。
目錄的大小計算,我們知道檔案的大小可以由filesize取得,但是php中沒有專門計算目錄大小的函數。當然php中有計算硬碟大小的函數disk_total_space(計算硬碟總空間)、disk_free_space(計算硬碟可用空間),但是我試了下disk_free_space,貌似計算得不對。正因為有filesize計算檔案的大小,因此,需要用到遞迴,當是目錄時,進去繼續計運算元目錄的大小,如果是檔案,擷取到檔案大小並加上返回,代碼如下:
複製代碼 代碼如下:
<?php
// 目錄大小計算
function getDirSize($dirpath){
$size = 0;
if(false != ($handle = opendir($dirpath))){
while(false != ($file = readdir($handle))){
if($file == '.' || $file == '..') //注意過濾目錄中內建的點和點點
continue;
$filepath = $dirpath.'/'.$file; // 前面要接上路徑
if(is_file($filepath)){ // 是檔案計算大小
$size += filesize($filepath);
}
else if(is_dir($filepath)){ // 是目錄繼續計算該目錄下的檔案
$size += getDirSize($filepath);
}
else{
$size += 0;
}
}
closedir($handle);
}
return $size;
}
$dirsize = 'F:/size';
$size = getDirSize($dirsize);
echo 'dir size: '.getSize(null, $size).'
'; // 調用前面的資料處理函數
我在F盤建了個size檔案,隨便弄了點子目錄和文檔,效果如下,左邊是程式求得,右邊是右鍵查看檔案夾屬性得到的,用以對比。
目錄的建立和刪除,主要用到,mkdir:建立一個目錄,rmdir:刪除一個非空目錄,注意只能是非空,代碼如下:
複製代碼 代碼如下:
<?php
// 目錄的建立和刪除
$newDirPath = 'F:/newDir';
if(true == @mkdir($newDirPath, 0777, true)){ // 加@是因為檔案已存在時php本身可能會拋出一個warning
echo 'create directory '.$newDirPath.' successfully
';
}
else{
if(file_exists($newDirPath))
echo 'directory '.$newDirPath.' has existed
';
else
echo 'create directory '.$newDirPath.' failed
';
}
if(true == @rmdir('F:/aaa')) //只能刪除非空目錄,如果刪除不存在的目錄自動拋出warning
echo 'remove successfully
';
那麼問題來了,如果要刪除一個非空目錄咋辦,又得自己寫了,思想仍然是遞迴,因為php只提供了刪除檔案函數unlink,所以在刪除一個目錄時,先opendir,再進入,如果是檔案直接刪除,如果是目錄,繼續進入使用該方法處理,當然還可已返回一個bool變數表示刪除是否成功,代碼如下:
複製代碼 代碼如下:
<?php
// 刪除檔案 unlink
// 刪除目錄中的內容,然後刪除該目錄
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;
}
在這裡不得不說遇到的一個大坑,就是 . 和 .. 這兩個鬼玩意兒(點和點點),在作業系統中的每一個檔案夾下邊,都會有 . 和 .. ,它們表示目前的目錄和目前的目錄的上級目錄,可惡的是前面在讀取目錄時居然沒顯示,導致遞迴函式成了死迴圈,因為 . 和 .. 在每一個目錄的最前面,必然會先讀到它倆,若不過濾,首先讀到 . ,它表示本目錄,然後又遞迴進入本目錄...這倆是作業系統下面的預設有的,它們是本目錄與上級目錄的串連符。
通過計算目錄的大小和刪除非空目錄的代碼,寫複製和剪下目錄就非常容易,非常相似的遞迴思想,需要用到複製檔案函數copy,檔案移動函數rename,這個挺有趣,rename,字面上是重新命名,但是重新命名到另一個目錄中不就是剪下了麼-_-
3.檔案讀寫
php的某些檔案讀取操作跟C語言非常像,所以也比較簡單,步驟就是先開啟檔案擷取控制代碼,檢查錯誤,然後讀寫處理,然後關閉,養成開啟處理完後就關閉的好習慣,記得在C語言中的檔案不關閉的話,開啟兩次是會報錯滴,不知道記錯沒,所以嚴格點的程式都有非常多的處理,比如先驗證檔案存在,然後驗證可讀可寫性,然後先關閉一下,然後再開啟,開啟時還得再檢查開啟錯了沒......在開啟檔案時,就要選擇開啟檔案的模式,它決定了我們讀還是寫檔案,當然是對需要這樣操作的函數有用。
寫檔案,寫檔案函數只有fwrite、fputs、file_put_contents少數幾個,其中fwrite與fputs效果一樣,file_put_contents是一次性向檔案寫入一些內容,它就不需要指定開啟模式,同時它也可以是附加或者覆蓋現有檔案內容,比如:
複製代碼 代碼如下:
<?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); // 附加內容
讀檔案,讀檔案的函數多些,有fread(讀取指定個位元組)、fgetc(讀取一個)、fgets(讀取一行)、file(全部讀取,按行分配到一個數組中返回)、file_get_contents(預設讀取全部返回字串)、readfile(直接將檔案中內容輸出到緩衝,效果就是直接在瀏覽器上輸出),伴隨著fread、fget、fgets運行,檔案指標會自動往後走。因此連續讀最好是迴圈控制。讀到檔案末尾怎麼辦,EOF標識指示到達檔案末尾,最好用feof檢測是否到檔案末尾。不多說,看代碼:
複製代碼 代碼如下:
<?php
// fread讀取
function readSome($filepath){
if(($handle = @fopen($filepath, 'r')) == true){
while(!feof($handle)){ // 判斷是否到達檔案末尾
$str = fread($handle, 10); // fread讀取時,檔案指標自動向後移動
echo $str.'
';
}
}
}
如果想要讀取方式更靈活,就要配合fseek、rewind使用,它們可以移動檔案指標到具體位置,fseek十分靈活,可以直接移到開頭或末尾,或從當前位置往前或後移動,讀取想要的內容,ftell還可告知當前位置,比如:
複製代碼 代碼如下:
<?php
function readFun($filepath){
if(($handle = @fopen($filepath, 'r')) != false){
echo 'current position: '.ftell($handle).'
'; // 輸出檔案當前檔案指標位置,以位元組算,0表示開頭
$str = fread($handle, 3); // 讀取3個位元組,同時指標自動後移3個位元組
echo 'read content: '.$str.'
';
echo 'current position: '.ftell($handle).'
';
fseek($handle, 5, SEEK_CUR); // 將檔案指標從當前位置後移5個位元組
echo 'current position: '.ftell($handle).'
';
$str = fread($handle, 5);
echo 'read content: '.$str.'
';
echo 'current position: '.ftell($handle).'
';
rewind($handle); // 返迴文件開頭
echo 'current position: '.ftell($handle).'
';
fseek($handle, 0, SEEK_END); // 移到檔案末尾
echo 'current position: '.ftell($handle).'
';
fclose($handle); // 關閉檔案
}
}
比如我現在使用該方法讀取一個寫有從a到z的文字檔,看看效果:
以上就是php關於目錄檔案操作的全部內容了,也是個人的一份理解記錄吧,希望對大家有所協助
http://www.bkjia.com/PHPjc/909339.htmlwww.bkjia.comtruehttp://www.bkjia.com/PHPjc/909339.htmlTechArticle詳談PHP檔案目錄基礎操作, 我們知道,臨時聲明的變數是儲存在記憶體中的,即便是靜態變數,在指令碼運行完畢後也會被釋放掉,so,想長久...