什么是目录
目录是一种特殊的文件,它的内容是文件和目录的名字。它们都包含很多记录,每个记录的格式由统一的标准定义。每条记录的内容代表一个文件或目录。与普通文件不同的是,目录文件永远不会空,每个目录都至少包含两个特殊的项"."和"..",其中". "表示当前目录,".."表示上一级目录。
什么函数可以读目录?
在联机帮助中根据关键字direct 来查找答案,用grep 过滤出那些包含read 的主题:
$ man -k direct | grep read
readdir (2) - read directory entry
readdir (3) - read a directory
readdir_r (3) - read a directory
目录是文件的列表,更确切地说,是记录的序列,每条记录对应一个文件或子目录。通过readdir 来读取目录中的记录, readdir 返回一个指向目录的当前记录的指针,记录的类型是struct dirent ,这个结构定义在/ usr/ include/ dirent. h
用stat 得到文件信息
磁盘上的文件有很多属性,如文件大小、文件所有者的ID 等。如果需要得到文件属性,进程可以定义一个结构struct stat ,然后调用stat ,告诉内核把文件属性存放到这个结构中。
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
int stat(const char *pathname, struct stat *statbuf);
ls程序代码
#include <stdio.h>
#include <sys/types.h>
#include <dirent.h>
#include <sys/stat.h>
void do_ls(char[]);
void dostat(char *);
void show_file_info(char *, struct stat *);
void mode_to_letters(int, char[]);
char *uid_to_name(uid_t);
char *gid_to_name(gid_t);
int main(int ac, char *av[])
{
if(ac == 1){ /*判断是否有参数,没有就传当前目录 . */
do_ls(".");
}else{
while(--ac){
printf("%s:\n", *++av);
do_ls(*av);
}
}
}
void do_ls(char dirname[])
{
DIR *dir_ptr;
struct dirent *direntp;
if((dir_ptr = opendir(dirname)) == NULL){ /*打开目录*/
fprintf(stderr, "ls1:cannot open %s\n", dirname);
}else{
while((direntp = readdir(dir_ptr)) != NULL){ /*读取目录*/
dostat(direntp->d_name);
}
closedir(dir_ptr); /*关闭目录*/
}
}
void dostat(char *filename) /*stat结构包含文件大小,时间等信息*/
{
struct stat info;
if(stat(filename, &info) == -1){
perror(filename);
}else{
show_file_info(filename, &info);
}
}
void show_file_info(char *filename, struct stat *info_p) /*文件详细打印函数*/
{
char *uid_to_name(), *ctime(), *gid_to_name(), *filemode();
void mode_to_letters();
char modestr[11];
mode_to_letters(info_p->st_mode, modestr);
printf("%s", modestr); /*文件类型和权限*/
printf("%4d ", (int) info_p->st_nlink); /*链接数*/
printf("%-8s", uid_to_name(info_p->st_uid)); /*用户*/
printf("%-8s", gid_to_name(info_p->st_gid)); /*用户组*/
printf("%8ld ", (long)info_p->st_size); /*文件大小*/
printf("%.12s ", 4 + ctime(&info_p->st_mtime)); /*时间*/
printf("%s\n", filename); /*文件名*/
}
void mode_to_letters(int mode, char str[])
{
strcpy(str, "----------");
if(S_ISDIR(mode)) str[0] = 'd'; /*目录*/
if(S_ISCHR(mode)) str[0] = 'c'; /*字符设备*/
if(S_ISBLK(mode)) str[0] = 'b'; /*块设备*/
if(mode & S_IRUSR) str[1] = 'r'; /*用户权限 r w x*/
if(mode & S_IWUSR) str[2] = 'w';
if(mode & S_IXUSR) str[3] = 'x';
if(mode & S_IRGRP) str[4] = 'r'; /*用户组权限 r w x*/
if(mode & S_IWGRP) str[5] = 'w';
if(mode & S_IXGRP) str[6] = 'x';
if(mode & S_IROTH) str[7] = 'r'; /*其它权限 r w x*/
if(mode & S_IWOTH) str[8] = 'w';
if(mode & S_IXOTH) str[9] = 'x';
}
#include <pwd.h>
char *uid_to_name(uid_t uid) /*用户id 转用户名*/
{
struct passwd *getpwuid(), *pw_ptr;
static char numstr[10];
if((pw_ptr = getpwuid(uid)) == NULL){
sprintf(numstr, "%d", uid);
return numstr;
}else{
return pw_ptr->pw_name;
}
}
#include <grp.h>
char *gid_to_name(gid_t gid) /*用户组gid 转组名*/
{
struct group *getgrgid(), *grp_ptr;
static char numstr[10];
if((grp_ptr = getgrgid(gid)) == NULL){
sprintf(numstr, "%d", gid);
return numstr;
}else{
return grp_ptr->gr_name;
}
}
测试
$ gcc -o ls2 ls.c
$ ./ls2
-rwxr-xr-x 1 root root 1178 Mar 11 12:30 test.c