功能说明
递归进入当前目录及其子目录,遇到普通文件就对计数加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;
}
文章评论