目錄
PHP之文件目录基础操作
首頁 後端開發 php教程 PHP之文件目录基础操作_PHP教程

PHP之文件目录基础操作_PHP教程

Jul 13, 2016 am 10:14 AM
基礎 文件 目錄

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:获取文件的一些基本信息,返回一个索引与关联混合数组。
 
     比如,可以这样判断文件类型:
 
复制代码
    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返回的是以字节为单位的数据,如果是大文件数字或很大,可以对数字先处理一下,代码如下
 
复制代码
    // 处理文件大小
    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';
          }
     }
复制代码
    现在综合来获取一下文件信息,代码如下:
 
复制代码
    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语言读取文件时也是这样,打开就有关闭。以我的机子举例:
 
复制代码
    // 目录的读取
    $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计算文件的大小,因此,需要用到递归,当是目录时,进去继续计算子目录的大小,如果是文件,获取到文件大小并加上返回,代码如下:
 
复制代码
    // 目录大小计算
     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:删除一个非空目录,注意只能是非空,代码如下:
 
复制代码
    // 目录的新建和删除
    $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变量表示删除是否成功,代码如下:
 
复制代码
    // 删除文件  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是一次性向文件写入一些内容,它就不需要指定打开模式,同时它也可以是附加或者覆盖现有文件内容,比如:
 
复制代码
    // 写  fwrite(别名fputs)
    $filepath = 'F:/10m.txt';
    function writeSome($filepath){
        if(($handle = fopen($filepath, 'r+')) == true){
             for($i=0; $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检测是否到文件末尾。不多说,看代码:
 
复制代码
    // fread读取
    function readSome($filepath){
        if(($handle = @fopen($filepath, 'r')) == true){
            while(!feof($handle)){            // 判断是否到达文件末尾
                $str = fread($handle, 10);    // fread读取时,文件指针自动向后移动
                echo $str.'
';
            }
         }
    }
复制代码
      如果想要读取方式更灵活,就要配合fseek、rewind使用,它们可以移动文件指针到具体位置,fseek十分灵活,可以直接移到开头或末尾,或从当前位置往前或后移动,读取想要的内容,ftell还可告知当前位置,比如:
 
复制代码
     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);  // 关闭文件
         }
     }

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/909451.htmlTechArticlePHP之文件目录基础操作 我们知道,临时声明的变量是保存在内存中的,即便是静态变量,在脚本运行完毕后也会被释放掉,so,想长久保存...
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1655
14
CakePHP 教程
1414
52
Laravel 教程
1307
25
PHP教程
1254
29
C# 教程
1228
24
可以刪除Tmp格式檔案嗎? 可以刪除Tmp格式檔案嗎? Feb 24, 2024 pm 04:33 PM

Tmp格式檔案是一種暫存檔案格式,通常由電腦系統或程式在執行過程中產生。這些文件的目的是儲存臨時數據,以幫助程式正常運行或提高效能。一旦程式執行完成或電腦重啟,這些tmp檔案往往就沒有了存在的必要性。所以,對於Tmp格式檔案來說,它們本質上是可以刪除的。而且,刪除這些tmp檔案能夠釋放硬碟空間,確保電腦的正常運作。但是,在刪除Tmp格式檔案之前,我們需

出現0x80004005錯誤代碼怎麼辦 小編教你0x80004005錯誤代碼解決方法 出現0x80004005錯誤代碼怎麼辦 小編教你0x80004005錯誤代碼解決方法 Mar 21, 2024 pm 09:17 PM

在電腦中刪除或解壓縮資料夾,時有時會彈出提示對話框“錯誤0x80004005:未指定錯誤”,如果遇到這中情況應該怎麼解決呢?提示錯誤碼0x80004005的原因其實很多,但大部分因為病毒導致,我們可以重新註冊dll來解決問題,下面,小編給大夥講解0x80004005錯誤代碼處理經驗。有使用者在使用電腦時出現錯誤代碼0X80004005的提示,0x80004005錯誤主要是由於電腦沒有正確註冊某些動態連結庫文件,或電腦與Internet之間存在不允許的HTTPS連接防火牆所引起。那麼如何

斜線和反斜線在檔案路徑中的不同使用 斜線和反斜線在檔案路徑中的不同使用 Feb 26, 2024 pm 04:36 PM

檔案路徑是作業系統中用於識別和定位檔案或資料夾的字串。在檔案路徑中,常見的有兩種符號分隔路徑,即正斜線(/)和反斜線()。這兩個符號在不同的作業系統中有不同的使用方式和意義。正斜線(/)是Unix和Linux系統中常用的路徑分隔符號。在這些系統中,檔案路徑是以根目錄(/)為起始點,每個目錄之間使用正斜線進行分隔。例如,路徑/home/user/Docume

夸克網盤的檔案怎麼轉移到百度網盤? 夸克網盤的檔案怎麼轉移到百度網盤? Mar 14, 2024 pm 02:07 PM

  夸克網盤和百度網盤都是現在最常用的儲存文件的網盤軟體,如果想要將夸克網盤內的文件保存到百度網盤,要怎麼操作呢?本期小編整理了夸克網盤電腦端的檔案轉移到百度網盤的教學步驟,一起來看看是怎麼操作吧。  夸克網盤的檔案怎麼存到百度網盤?要將夸克網盤的文件轉移到百度網盤,首先需在夸克網盤下載所需文件,然後在百度網盤用戶端中選擇目標資料夾並開啟。接著,將夸克網盤中下載的檔案拖放到百度網盤用戶端開啟的資料夾中,或使用上傳功能將檔案新增至百度網盤。確保上傳完成後在百度網盤中查看檔案是否已成功轉移。這樣就

hiberfil.sys是什麼檔案? hiberfil.sys可以刪除嗎? hiberfil.sys是什麼檔案? hiberfil.sys可以刪除嗎? Mar 15, 2024 am 09:49 AM

  最近有很多網友問小編,hiberfil.sys是什麼文件? hiberfil.sys佔用了大量的C碟空間可以刪除嗎?小編可以告訴大家hiberfil.sys檔是可以刪除的。下面就來看看詳細的內容。 hiberfil.sys是Windows系統中的隱藏文件,也是系統休眠文件。通常儲存在C盤根目錄下,其大小與系統安裝記憶體大小相當。這個檔案在電腦休眠時被使用,其中包含了當前系統的記憶體數據,以便在恢復時快速恢復到先前的狀態。由於其大小與記憶體容量相等,因此它可能會佔用較大的硬碟空間。  hiber

Linux系統查看log日誌指令詳解! Linux系統查看log日誌指令詳解! Mar 06, 2024 pm 03:55 PM

在Linux系統中,可以使用下列指令來查看日誌檔案的內容:tail指令:tail指令用來顯示日誌檔案的末尾內容。它是查看最新日誌資訊的常用命令。 tail[選項][檔案名稱]常用的選項包括:-n:指定要顯示的行數,預設為10行。 -f:即時監視文件內容,並在文件更新時自動顯示新的內容。範例:tail-n20logfile.txt#顯示logfile.txt檔案的最後20行內容tail-flogfile.txt#即時監視logfile.txt檔案的更新內容head指令:head指令用於顯示記錄檔的開頭

MySQL中.ibd檔的作用詳解及相關注意事項 MySQL中.ibd檔的作用詳解及相關注意事項 Mar 15, 2024 am 08:00 AM

MySQL中.ibd檔案的作用詳解及相關注意事項MySQL是一種流行的關聯式資料庫管理系統,資料庫中的資料儲存在不同的檔案中。其中,.ibd檔案是InnoDB儲存引擎中的資料文件,用於儲存表格中的資料和索引。本文將對MySQL中.ibd檔案的作用進行詳細解析,並提供相關程式碼範例以幫助讀者更好地理解。一、.ibd檔的作用:儲存資料:.ibd檔是InnoDB存

建立並執行Linux'.a”文件 建立並執行Linux'.a”文件 Mar 20, 2024 pm 04:46 PM

在Linux作業系統中處理檔案需要使用各種命令和技術,使開發人員能夠有效率地建立和執行檔案、程式碼、程式、腳本和其他東西。在Linux環境中,擴展名為”.a”的檔案作為靜態庫具有重要的重要性。這些程式庫在軟體開發中發揮重要作用,允許開發人員有效地管理和共享多個程式的公共功能。對於Linux環境中的有效軟體開發,了解如何建立和運行「.a」檔案至關重要。本文將介紹如何全面安裝和設定Linux「.a」文件,讓我們一起探索Linux「.a」文件的定義、用途、結構,以及建立和執行它的方法。什麼是L

See all articles