美文网首页
文件I/O(二)

文件I/O(二)

作者: mnikn | 来源:发表于2017-03-25 11:19 被阅读9次

    本文转载自实验楼:文件I/O(二)

    stat 的使用

    Linux有个命令,ls -l,效果如下:

    $ ls -l
    -rw-rw-r-- 1 shiyanlou shiyanlou 978 Sep 19 02:13 hello.c
    

    这个命令能显示文件的类型、操作权限、硬链接数量、属主、所属组、大小、修改时间、文件名。它是怎么获得这些信息,这一节我们将拨开迷雾。

    stat 的基本使用

    系统调用stat的作用是获取文件的各个属性。
    需要包含的头文件:

    • <sys/types.h>
    • <sys/stat.h>
    • <unistd.h>

    函数原型:

    int stat(const char \* path, struct stat \* buf)
    

    功能:查看文件或目录属性。将参数path所指的文件的属性,复制到参数buf所指的结构中。
    参数:

    • path:要查看属性的文件或目录的全路径名称。
    • buf:指向用于存放属性的结构体。stat成功调用后,buf的各个字段将存放各个属性。
    • struct stat是系统头文件中定义的结构体,定义如下:
    struct stat {
        dev_t       st_dev;
        ino_t       st_ino;
        mode_t      st_mode;
        nlink_t     st_nlink;
        uid_t       st_uid;
        gid_t       st_gid;
        dev_t       st_rdev;
        off_t       st_size;
        blksize_t   st_blksize;
        blkcnt_t    st_blocks;
        time_t      st_atime;
        time_t      st_mtime;
        time_t      st_ctime;
    };
    
    • st_ino:节点号
    • st_mode:文件类型和文件访问权限被编码在该字段中
    • st_nlink:硬连接数
    • st_uid:属主的用户ID
    • st_gid:所属组的组ID
    • st_rdev:设备文件的主、次设备号编码在该字段中
    • st_size:文件的大小
    • st_mtime:文件最后被修改时间
      返回值:成功返回0;失败返回-1
    #include <stdio.h>
    #include <stdlib.h>
    #include <sys/stat.h>
    #include <sys/types.h>
    #include <unistd.h>
    
    int main(int argc, char **argv)
    {
        struct stat     buf;
        if(argc != 2) { 
            printf("Usage: stat <pathname>"); 
            exit(-1); 
        }
        if(stat(argv[1], &buf) != 0) { 
            printf("stat error."); 
            exit(-1); 
        }
        printf("#i-node:    %ld\n", buf.st_ino);
        printf("#link:      %d\n", buf.st_nlink);
        printf("UID:        %d\n", buf.st_uid);
        printf("GID:        %d\n", buf.st_gid);
        printf("Size        %ld\n", buf.st_size);
        exit(0);
    }
    
    文件类型的判定

    上一小节中struct stat中有个字段为st_mode,可用来获取文件类型和文件访问权限,我们将陆续学到从该字段解码我们需要的文件信息
    st_mode中文件类型宏定义

    宏定义 文件类型
    S_ISREG() 普通文件
    S_ISDIR() 目录文件
    S_ISCHR() 字符设备文件
    S_ISBLK() 块设备文件
    S_ISFIFO() 有名管道文件
    S_ISLNK() 软连接(符号链接)文件
    S_ISSOCK() 套接字文件

    我们修改上面的例子:

    #include <stdio.h>
    #include <stdlib.h>
    #include <sys/stat.h>
    #include <sys/types.h>
    #include <unistd.h>
    
    int main(int argc, char **argv)
    {
        struct stat buf;
        char * file_mode;
        if(argc != 2) {
            printf("Usage: stat <pathname>\n"); 
            exit(-1); 
        }
        if(stat(argv[1], &buf) != 0) {
            printf("stat error.\n"); 
            exit(-1); 
        }
        if (S_ISREG(buf.st_mode))
            file_mode = "-";
        else if (S_ISDIR(buf.st_mode))
            file_mode = "d";
        else if (S_ISCHR(buf.st_mode))
            file_mode = "c";
        else if(S_ISBLK(buf.st_mode))
            file_mode = "b";
        printf("#i-node:    %ld\n", buf.st_ino);
        printf("#link:      %d\n", buf.st_nlink);
        printf("UID:        %d\n", buf.st_uid);
        printf("GID:        %d\n", buf.st_gid);
        printf("Size        %ld\n", buf.st_size);
        printf("mode: %s\n", file_mode);
        exit(0);
    }
    
    文件权限的判定

    文件类型与许可设定被一起编码在st_mode字段中,同上面一样,我们也需要一组由系统提供的宏来完成解码。

    宏定义 文件类型
    S_ISUID 执行时,设置用户ID
    S_ISGID 执行时,设置组ID
    S_ISVTX 保存正文
    S_IRWXU 拥有者的读、写和执行权限
    S_IRUSR 拥有者的读权限
    S_IWUSR 拥有者的写权限
    S_IXUSR 拥有者的执行权限
    S_IRWXG 用户组的读、写和执行权限
    S_IRGRP 用户组的读权限
    S_IWGRP 用户组的写权限
    S_IRWXO 其它读、写、执行权限
    S_IROTH 其它读权限
    S_IWOTH 其它写权限
    S_IXOTH 其它执行权限

    目录操作

    当目标是目录而不是文件的时候,ls -l的结果会显示目录下所有子条目的信息,怎么去遍历整个目录呢?

    打开目录

    需要包含的头文件:

    • <sys/types.h>
    • <dirent.h>

    函数原型:

    DIR * opendir(const char * name)
    

    功能:opendir()用来打开参数name指定的目录,并返回DIR *形态的目录流
    返回值:成功返回目录流;失败返回NULL

    读取目录

    函数原型:

    struct dirent * readdir(DIR * dir)
    

    功能:readdir()返回参数dir目录流的下一个子条目(子目录或子文件)
    返回值: 成功返回结构体指向的指针,错误或以读完目录,返回NULL
    函数执行成功返回的结构体原型如下:

    struct dirent {
       ino_t   d_ino;
       off_t   d_off;
       unsigned short  d_reclen;
       unsigned char   d_type;
       char    d_name[256];
    };
    

    其中 d_name字段,是存放子条目的名称

    关闭目录

    函数原型:

    int closedir(DIR * dir)
    

    功能:closedir()关闭dir所指的目录流
    返回值:成功返回0;失败返回-1,错误原因在errno中

    我们来学习一个综合的例子吧:

    #include <stdio.h>
    #include <stdlib.h>
    #include <dirent.h>
    int main(int argc, char *argv[])
    {
        DIR *dp;
        struct dirent *entp;
        if (argc != 2) {
            printf("usage: showdir dirname\n");
            exit(0);
        }
        if ((dp = opendir(argv[1])) == NULL) {
            perror("opendir");
            exit(-1);
        }
        while ((entp = readdir(dp)) != NULL)
            printf("%s\n", entp->d_name);
    
        closedir(dp);
        return 0;
    }
    

    相关文章

      网友评论

          本文标题:文件I/O(二)

          本文链接:https://www.haomeiwen.com/subject/ivxcottx.html