6

我使用 opendir() 打开一个目录,然后使用 readdir() 和 lstat() 来获取该目录中每个文件的统计信息。在此联机帮助页之后,我编写了无法按预期工作的代码。它确实列出了当前目录中的所有文件,但无论文件是常规文件、符号链接还是目录,它都不会打印出来。

#include <sys/stat.h>
#include <sys/types.h>
#include <dirent.h>
#include <stdio.h>

void main(){

    char* folder=".";                                     //folder to open

    DIR* dir_p;
    struct dirent* dir_element;
    struct stat file_info;

    // open directory
    dir_p=opendir(folder);

    // show some info for each file in given directory
    while(dir_element = readdir(dir_p)){

        lstat(dir_element->d_name, &file_info);          //getting a file stats

        puts(dir_element->d_name);                       // show current filename
        printf("file mode: %d\n", file_info.st_mode);

        // print what kind of file we are dealing with
        if (file_info.st_mode == S_IFDIR) puts("|| directory");
        if (file_info.st_mode == S_IFREG) puts("|| regular file");
        if (file_info.st_mode == S_IFLNK) puts("|| symbolic link");
    }

}
4

3 回答 3

6

我知道那是几年后的事了,但对于后代你做错了:
@alk是对的st_mode字段包含更多信息,例如文件类型、文件权限等
要提取文件类型,您按位执行st_mode字段和文件类型掩码S_IFMT。然后检查你想要的结果。这就是@Ernest Friedman-Hill提到的宏所做的。一个 swicth 更适合全面检查,即

对于一个简单的情况:

     if ((file_info.st_mode & S_IFMT)==S_IFDIR) puts("|| directory");

进行全面检查:

       struct stat st;
       ...

      switch (st.st_mode & S_IFMT) {
        case S_IFREG:  
            puts("|| regular file");
            break;
        case S_IFDIR:
            puts("|| directory");
            break;
        case S_IFCHR:        
            puts("|| character device");
            break;
        case S_IFBLK:        
            puts("|| block device");
            break;
        case S_IFLNK: 
            puts("|| symbolic link");
            break;
        case S_IFIFO: 
            puts("|| pipe");    
            break;
        case S_IFSOCK:
            puts("|| socket");
            break;
        default:
            puts("|| unknown"); 
     }
于 2016-03-14T00:34:06.177 回答
5

有一组宏要解释st_mode,这比你想象的要复杂。使用它们而不是直接探测该字段:

if (S_ISREG(file_info.st_mode))
    // file is a regular file
else if (S_ISLNK(file_info.st_mode))
    // ...

还有S_ISDIR,S_ISSOCK等等。例如,请参阅此处了解信息。

于 2011-10-06T12:42:22.777 回答
2

模式携带大量信息。

尝试以下类型的测试:

if (S_ISDIR(file_info.st_mode))  puts("|| directory");
于 2011-10-06T12:37:00.883 回答