递归读目录获取普通文件个数

功能说明

递归进入当前目录及其子目录,遇到普通文件就对计数加1,遇到目录就进入目录,但并不做加1操作,最后输出普通文件个数。

使用的函数

1.opendir()函数:打开一个目录
2.readdir()函数:读目录。这个函数为核心,判断当前目录是否是普通文件,根据其返回的结构体中的d_type来进行判断。
3.closedir()函数:关闭目录。

使用到的头文件

对每个函数使用man命令查看帮助,或者在.c文件中将光标放在该函数名上,按shift+k也可以查看帮助。

#include <sys/types.h>//opendir、closedir使用
#include <dirent.h>//opendir、closedir使用
#include <stdio.h>
#include <stdlib.h>

打开目录、遍历目录和关闭目录

打开目录
打开目录类似于打开文件,同样需要对是否出现错误进行判断,若出现错误则输出错误信息,然后退出。

//打开目录
    DIR* dir = NULL;
    dir = opendir(root);
    if(dir == NULL)
    {
        perror("Open dir error:");
        exit(1);
    }

遍历目录

由于readdir()函数的返回值是一个结构体,且当文件已经读完的时候返回NULL,因此可以利用是否返回NULL来作为判断循环结束的条件。

struct dirent* dr = NULL;
while((dr = readdir(dir))!= NULL)
    {
    ...
    }

关闭目录
直接执行关闭目录即可:

closeret = closedir(dir);

设计程序的时候的注意事项

  • 在读到普通目录的时候,我们要进入子目录,但是若读到.(当前目录)和..(上一级目录),我们就需要将它们排除在外。其原理就是使用strcmp()函数判断当前的名称是否是.或者..,若是则直接跳过,进行下一次循环。实现如下:
if(strcmp(dr->d_name, ".") == 0 || strcmp(dr->d_name, "..") == 0 )
        {
            continue;
        }
  • 读普通目录,进入子目录进行递归计数。子目录的文件名需要我们进行拼接,也就是使用sprintf()函数。代码如下:
if(dr->d_type == DT_DIR)
        {
            //递归读目录
            sprintf(path,"%s/%s",root,dr->d_name);
            total += getFileNum(path);
        }
  • 对于遇到普通文件,则直接对total进行+1,实现如下:
if(dr->d_type == DT_REG)
        {
            total++;
        }

完整代码

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

int getFileNum(char* root)
{
    //打开目录
    DIR* dir = NULL;
    dir = opendir(root);
    if(dir == NULL)
    {
        perror("Open dir error:");
        exit(1);
    }
    int total=0;
    //遍历当前目录
    struct dirent* dr = NULL;
    char path[1024] = {0};
    while((dr = readdir(dir))!= NULL)
    {
        //过滤.和..
        if(strcmp(dr->d_name, ".") == 0 || strcmp(dr->d_name, "..") == 0 )
        {
            continue;
        }
        //遇到目录,进入目录
        if(dr->d_type == DT_DIR)
        {
            //递归读目录
            sprintf(path,"%s/%s",root,dr->d_name);
            total += getFileNum(path);
        }
        //如果是普通文件
        if(dr->d_type == DT_REG)
        {
            total++;
        }
    }
    //关闭目录
    int closeret = closedir(dir);
    if(closeret == -1)
    {
        perror("Close File error:");
        exit(1);
    }
    return total;
}
int main(int argc, char* argv[])
{
    if(argc<2)
    {
        printf("Usage: ./filecount <dir>\n");
        exit(1);
    }

    int result = getFileNum(argv[1]);
    printf("目录 %s 共有普通文件 %d 个\n",argv[1],result);
    return 0;
}

实现结果

发表评论

您的电子邮箱地址不会被公开。 必填项已用*标注