操作录像
UNIX环境高级编程-1.4-列出一个目录中的所有文件
1. 所需资源下载地址
《UNIX环境高级编程(第三版)》一书中所有的源代码下载地址:
原书本章节的源码如下(本文第三节进行了修改):
#include "apue.h"
#include <dirent.h>
int main(int argc, char *argv[])
{
DIR *dp;
struct dirent *dirp;
if(argc != 2)
err_quit("useage: ls directory_name");
if((dp = opendir(argv[1])) == NULL)
err_sys("can't open %s", argv[1]);
while((dirp = readdir(dp)) != NULL)
printf("%s\n", dirp -> d_name);
closedir(dp);
exit(0);
}
2. 系统依赖
根据视频中的操作(1:28~3:30),下载源码,将apue.h和error.c复制到/usr/include目录中。然后编辑apue.h文件,在倒数第二行(即#endif前面一行),加上以下内容并保存。另外,请一定确保error.c是被英文的双引号包围的:
#include "error.c"
3. 关于结构体属性
《UNIX环境高级编程(第三版)》1.4章节,文中使用了以下结构体的字段,打印了文件夹名称:
dirp->d_name
这个结构体可以在命令行中输入以下命令找到:
man 5 dirent
如下图:
同时我们也可以看到对于d_type属性(文件类型)的定义,如下图:
通过百度查找这些文件类型的定义,得到如下注释:
DT_UNKNOWN = 0, //未知类型
DT_FIFO = 1, //管道
DT_CHR = 2, //字符设备
DT_DIR = 4, //目录
DT_BLK = 6, //块设备
DT_REG = 8, //常规文件
DT_LNK = 10, //符号链接
DT_SOCK = 12, //套接字
DT_WHT = 14 //链接
修改视频中的源程序代码如下(将文件名长度和文件类型一起打印出来):
#include "apue.h"
#include <dirent.h>
int main(int argc, char *argv[])
{
DIR *dp;
struct dirent *dirp;
if(argc != 2)
err_quit("useage: ls directory_name");
if((dp = opendir(argv[1])) == NULL)
err_sys("can't open %s", argv[1]);
while((dirp = readdir(dp)) != NULL)
printf("%s\t%d\t%d\n", dirp -> d_name, dirp -> d_namlen, dirp -> d_type);
closedir(dp);
exit(0);
}
重新编译执行效果如下图:
4. 总结
- dirent.h头文件中定义的opendir函数,得到一个DIR类型的指针,其实就是目录文件的文件句柄(不好理解的话,就试着理解为这个DIR变量存储的是目录文件存在于磁盘的物理位置,跟你家门牌号一样的东西)。
- 再通过把DIR丢给readdir函数,让readdir函数去查询该目录文件的详情,查询结果保存在一个dirent类型的结构体当中。
- 如果结构体不为NULL,则代表要查询的目录文件或目录下的子文件是存在的,这时候我们可以使用printf函数将当前查询到的目录文件信息,或当前目录下的子文件的信息打印出来,然后DIR指针偏移到下一个地址(就是敲完你家门,该敲下一家的门了),如果下一户的门敲开以后,你说还是你家,只不过是个侧门,那while((dirp = readdir(dp)) != NULL)语句又成立,继续打印文件信息,因为目录下的子文件也是目录的内容,可以理解为你家侧门也是你家的东西。
- 如果敲门敲到最后,敲到你邻居家了readdir函数就会返回NULL,此时结束循环,然后把DIR地址丢给函数closedir,释放文件句柄,这个closedir可以简单理解为,opendir的时候在你家门口立了个牌子,上书“内有贵宾来访,恕不接待”,所有来你家拜访的人都得在门口排队等着,closedir函数的作用是,把这个牌子摘掉,以便其他人来访。