本文转载自实验楼:文件I/O(二)
stat 的使用
Linux有个命令,ls -l,效果如下:
$ ls -l
-rw-rw-r-- 1 shiyanlou shiyanlou 978 Sep 19 02:13 hello.c
这个命令能显示文件的类型、操作权限、硬链接数量、属主、所属组、大小、修改时间、文件名。它是怎么获得这些信息,这一节我们将拨开迷雾。
stat 的基本使用
系统调用stat的作用是获取文件的各个属性。
需要包含的头文件:
- <sys/types.h>
- <sys/stat.h>
- <unistd.h>
函数原型:
int stat(const char \* path, struct stat \* buf)
功能:查看文件或目录属性。将参数path所指的文件的属性,复制到参数buf所指的结构中。
参数:
- path:要查看属性的文件或目录的全路径名称。
- buf:指向用于存放属性的结构体。stat成功调用后,buf的各个字段将存放各个属性。
- struct stat是系统头文件中定义的结构体,定义如下:
struct stat {
dev_t st_dev;
ino_t st_ino;
mode_t st_mode;
nlink_t st_nlink;
uid_t st_uid;
gid_t st_gid;
dev_t st_rdev;
off_t st_size;
blksize_t st_blksize;
blkcnt_t st_blocks;
time_t st_atime;
time_t st_mtime;
time_t st_ctime;
};
- st_ino:节点号
- st_mode:文件类型和文件访问权限被编码在该字段中
- st_nlink:硬连接数
- st_uid:属主的用户ID
- st_gid:所属组的组ID
- st_rdev:设备文件的主、次设备号编码在该字段中
- st_size:文件的大小
- st_mtime:文件最后被修改时间
返回值:成功返回0;失败返回-1
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
int main(int argc, char **argv)
{
struct stat buf;
if(argc != 2) {
printf("Usage: stat <pathname>");
exit(-1);
}
if(stat(argv[1], &buf) != 0) {
printf("stat error.");
exit(-1);
}
printf("#i-node: %ld\n", buf.st_ino);
printf("#link: %d\n", buf.st_nlink);
printf("UID: %d\n", buf.st_uid);
printf("GID: %d\n", buf.st_gid);
printf("Size %ld\n", buf.st_size);
exit(0);
}
文件类型的判定
上一小节中struct stat中有个字段为st_mode,可用来获取文件类型和文件访问权限,我们将陆续学到从该字段解码我们需要的文件信息
。st_mode中文件类型宏定义:
宏定义 | 文件类型 |
---|---|
S_ISREG() | 普通文件 |
S_ISDIR() | 目录文件 |
S_ISCHR() | 字符设备文件 |
S_ISBLK() | 块设备文件 |
S_ISFIFO() | 有名管道文件 |
S_ISLNK() | 软连接(符号链接)文件 |
S_ISSOCK() | 套接字文件 |
我们修改上面的例子:
#include <stdio.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
int main(int argc, char **argv)
{
struct stat buf;
char * file_mode;
if(argc != 2) {
printf("Usage: stat <pathname>\n");
exit(-1);
}
if(stat(argv[1], &buf) != 0) {
printf("stat error.\n");
exit(-1);
}
if (S_ISREG(buf.st_mode))
file_mode = "-";
else if (S_ISDIR(buf.st_mode))
file_mode = "d";
else if (S_ISCHR(buf.st_mode))
file_mode = "c";
else if(S_ISBLK(buf.st_mode))
file_mode = "b";
printf("#i-node: %ld\n", buf.st_ino);
printf("#link: %d\n", buf.st_nlink);
printf("UID: %d\n", buf.st_uid);
printf("GID: %d\n", buf.st_gid);
printf("Size %ld\n", buf.st_size);
printf("mode: %s\n", file_mode);
exit(0);
}
文件权限的判定
文件类型与许可设定被一起编码在st_mode字段中,同上面一样,我们也需要一组由系统提供的宏来完成解码。
宏定义 | 文件类型 |
---|---|
S_ISUID | 执行时,设置用户ID |
S_ISGID | 执行时,设置组ID |
S_ISVTX | 保存正文 |
S_IRWXU | 拥有者的读、写和执行权限 |
S_IRUSR | 拥有者的读权限 |
S_IWUSR | 拥有者的写权限 |
S_IXUSR | 拥有者的执行权限 |
S_IRWXG | 用户组的读、写和执行权限 |
S_IRGRP | 用户组的读权限 |
S_IWGRP | 用户组的写权限 |
S_IRWXO | 其它读、写、执行权限 |
S_IROTH | 其它读权限 |
S_IWOTH | 其它写权限 |
S_IXOTH | 其它执行权限 |
目录操作
当目标是目录而不是文件的时候,ls -l的结果会显示目录下所有子条目的信息,怎么去遍历整个目录呢?
打开目录
需要包含的头文件:
- <sys/types.h>
- <dirent.h>
函数原型:
DIR * opendir(const char * name)
功能:opendir()用来打开参数name指定的目录,并返回DIR *形态的目录流
返回值:成功返回目录流;失败返回NULL
读取目录
函数原型:
struct dirent * readdir(DIR * dir)
功能:readdir()返回参数dir目录流的下一个子条目(子目录或子文件)
返回值: 成功返回结构体指向的指针,错误或以读完目录,返回NULL
函数执行成功返回的结构体原型如下:
struct dirent {
ino_t d_ino;
off_t d_off;
unsigned short d_reclen;
unsigned char d_type;
char d_name[256];
};
其中 d_name字段,是存放子条目的名称
关闭目录
函数原型:
int closedir(DIR * dir)
功能:closedir()关闭dir所指的目录流
返回值:成功返回0;失败返回-1,错误原因在errno中
我们来学习一个综合的例子吧:
#include <stdio.h>
#include <stdlib.h>
#include <dirent.h>
int main(int argc, char *argv[])
{
DIR *dp;
struct dirent *entp;
if (argc != 2) {
printf("usage: showdir dirname\n");
exit(0);
}
if ((dp = opendir(argv[1])) == NULL) {
perror("opendir");
exit(-1);
}
while ((entp = readdir(dp)) != NULL)
printf("%s\n", entp->d_name);
closedir(dp);
return 0;
}
网友评论