2015年1月9日 星期五

Linux下DIR,dirent,stat等结构体详解

  
  DIR结构体类似于FILE,是一个内部结构,以下几个函数用这个内部结构保存当前正在被读取的目录的有关信息(摘自《UNIX环境高级编程(第二版)》)。函数 DIR *opendir(const char *pathname),即打开文件目录,返回的就是指向DIR结构体的指针,而该指针由以下几个函数使用:

首先说说DIR这一结构体,以下为DIR结构体的定义:
  1. struct __dirstream
  2.    {
  3.     void *__fd;
  4.     char *__data;
  5.     int __entry_data;
  6.     char *__ptr;
  7.     int __entry_ptr;
  8.     size_t __allocation;
  9.     size_t __size;
  10.     __libc_lock_define (, __lock)
  11.    };
  12. typedef struct __dirstream DIR;

DIR结构体类似于FILE,是一个内部结构,以下几个函数用这个内部结构保存当前正在被读取的目录的有关信息(摘自《UNIX环境高级编程(第二版)》)。函数 DIR *opendir(const char *pathname),即打开文件目录,返回的就是指向DIR结构体的指针,而该指针由以下几个函数使用:

  1. struct dirent *readdir(DIR *dp);
  2. void rewinddir(DIR *dp);
  3. int closedir(DIR *dp);
  4. long telldir(DIR *dp);
  5. void seekdir(DIR *dp,long loc);

关于DIR结构,我们知道这么多就可以了,没必要去再去研究他的结构成员。
接着是dirent结构体,首先我们要弄清楚目录文件(directory file)的概念:这种文件包含了其他文件的名字以及指向与这些文件有关的信息的指针(摘自《UNIX环境高级编程(第二版)》)。从定义能够看出,dirent不仅仅指向目录,还指向目录中的具体文件,readdir函数同样也读取目录下的文件,这就是证据。以下为dirent结构体的定义:

  1. struct dirent
  2. {
  3.   long d_ino; /* inode number 索引节点号 */
  4.     off_t d_off; /* offset to this dirent 在目录文件中的偏移 */
  5.     unsigned short d_reclen; /* length of this d_name 文件名长 */
  6.     unsigned char d_type; /* the type of d_name 文件类型 */
  7.     char d_name [NAME_MAX+1]; /* file name (null-terminated) 文件名,最长255字符 */
  8. }

从上述定义也能够看出来,dirent结构体存储的关于文件的信息很少,所以dirent同样也是起着一个索引的作用,如果想获得类似ls -l那种效果的文件信息,必须要靠stat函数了。
通过readdir函数读取到的文件名存储在结构体dirent的d_name成员中,而函数
int stat(const char *file_name, struct stat *buf);
的作用就是获取文件名为d_name的文件的详细信息,存储在stat结构体中。以下为stat结构体的定义:

  1. struct stat {
  2.         mode_t     st_mode;       //文件访问权限
  3.         ino_t      st_ino;       //索引节点号
  4.         dev_t      st_dev;        //文件使用的设备号
  5.         dev_t      st_rdev;       //设备文件的设备号
  6.         nlink_t    st_nlink;      //文件的硬连接数
  7.         uid_t      st_uid;        //所有者用户识别号
  8.         gid_t      st_gid;        //组识别号
  9.         off_t      st_size;       //以字节为单位的文件容量
  10.         time_t     st_atime;      //最后一次访问该文件的时间
  11.         time_t     st_mtime;      //最后一次修改该文件的时间
  12.         time_t     st_ctime;      //最后一次改变该文件状态的时间
  13.         blksize_t st_blksize;    //包含该文件的磁盘块的大小
  14.         blkcnt_t   st_blocks;     //该文件所占的磁盘块
  15.       };

这个记录的信息就很详细了吧,呵呵。
最后,总结一下,想要获取某目录下(比如a目下)b文件的详细信息,我们应该怎样做?
首先,我们使用opendir函数打开目录a,返回指向目录a的DIR结构体c。
接着,我们调用readdir(c)函数读取目录a下所有文件(包括目录),返回指向目录a下所有文件的dirent结构体d。
然后,我们遍历d,调用stat(d->name,stat *e)来获取每个文件的详细信息,存储在stat结构体e中。
总体就是这样一种逐步细化的过程,在这一过程中,三种结构体扮演着不同的角色。

以下範例為開啟某目錄後,搜尋目錄下所有".c"副檔名,此處是用指針偏移的方式來取得附檔名,另外也可以用函數fnmatch來實現。

#include<stdio.h>
#include<string.h>
#include<unistd.h>
#include<dirent.h>
#include<stdlib.h>

int main (){
    DIR *d;
    struct dirent *de;
    char * buf;

    d=opendir("/root/pratice/day5");
    if(d==NULL){printf("%m\n"),exit(-1);}

    while((de=readdir(d)))
    {
    int length=strlen(de->d_name);
    int ext=length-2;
    buf=strstr(de->d_name+ext,".c");
    if (buf){printf("%s\n",de->d_name);}
    }
    return 0;
}

沒有留言:

張貼留言