前言
学习linxu下的文件属性和目录的获取可以帮助我们更进一步的了解这个系统一切皆为文件的神奇,通过学习我们可以直到这些知识都是伴随着stat函数,xxxdir类型的函数和一些组和时间戳的概念来完成的,最后我们能完成一个类似于ls函数的功能
stat函数
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
int stat(const char *path, struct stat *buf);
功能:通过该函数可以获取指定文件的属性信息。
参数:path 要获取属性信息的文件路径+名称,可以是
所有文件不限制文件类型。
buf 用于获取文件信息后数据返回的指针
buf对应的结构体信息,详见man 2 stat
返回值:成功 0
失败 -1
其中stuct stat 结构体包含下列内容
struct stat {
dev_t st_dev; //文件的设备编号
ino_t st_ino; //节点
mode_t st_mode; //文件的类型和存取的权限
nlink_t st_nlink; //连到该文件的硬连接数目,刚建立的文件值为1
uid_t st_uid; //用户ID
gid_t st_gid; //组ID
dev_t st_rdev; //(设备类型)若此文件为设备文件,则为其设备编号
off_t st_size; //文件字节数(文件大小)
unsigned long st_blksize; //块大小(文件系统的I/O 缓冲区大小)
unsigned long st_blocks; //块数
time_t st_atime; //最后一次访问时间
time_t st_mtime; //最后一次修改时间
time_t st_ctime; //最后一次改变时间(指属性)
};
其中在st_mode 中包含有权限的宏定义和位图
补充 linxu下的7种文件类型
普通文件 -
目录文件 d
链接文件 l /dev/systty
字符设备文件 c /dev/tty
块设备文件 b /dev/sr0
管道文件 p ./fifo
套接字文件 s /dev/log
S_IFMT 0170000 文件类型的位遮罩
S_IFSOCK 0140000 scoket
S_IFLNK 0120000 符号连接
S_IFREG 0100000 一般文件
S_IFBLK 0060000 区块装置
S_IFDIR 0040000 目录
S_IFCHR 0020000 字符装置
S_IFIFO 0010000 先进先出
S_ISUID 04000 文件的(set user-id on execution)位
S_ISGID 02000 文件的(set group-id on execution)位
S_ISVTX 01000 文件的sticky位
S_IRUSR(S_IREAD) 00400 文件所有者具可读取权限
S_IWUSR(S_IWRITE)00200 文件所有者具可写入权限
S_IXUSR(S_IEXEC) 00100 文件所有者具可执行权限
S_IRGRP 00040 用户组具可读取权限
S_IWGRP 00020 用户组具可写入权限
S_IXGRP 00010 用户组具可执行权限
S_IROTH 00004 其他用户具可读取权限
S_IWOTH 00002 其他用户具可写入权限
S_IXOTH 00001 其他用户具可执行权限
复制代码
上述的文件类型在POSIX中定义了检查这些类型的宏定义:
复制代码
S_ISLNK (st_mode) 判断是否为符号连接
S_ISREG (st_mode) 是否为一般文件
S_ISDIR (st_mode) 是否为目录
S_ISCHR (st_mode) 是否为字符装置文件
S_ISBLK (s3e) 是否为先进先出
S_ISSOCK (st_mode) 是否为socket
struct stat buf; // stat 的基本使用,通过宏定义判断文件类型
int res=stat (pathname,&buf);
if (res <0)
{
printf ("error stat\n");
return -1;
}
if (S_ISDIR(buf.st_mode))
{
printf ("文件类型为d\n");
}
for (int i=8;i>=0;i--) // 通过位图来判断文件权限
{
if (buf.st_mode & (1<<i))
{
switch (i%3)
{
case 2:
printf ("r");
break;
case 1:
printf ("w");
break;
case 0:
printf ("x");
break;
}
}
// 通过宏来判断文件类型
if(buf.st_mode & S_IRUSR) printf("r"); else printf("-");
if(buf.st_mode & S_IWUSR) printf("w"); else printf("-");
if(buf.st_mode & S_IXUSR) printf("x"); else printf("-");
stat 通过文件路径获取属性,面对符号链接文件时获取的是
指向文件目标的属性
fstat 通过文件描述符获取属性
lstat 面对符号链接文件时获取的是符号链接文件的属性
文件目录操作
和标准io一样目录的操作会打开一个目录流,你可以从目录流中获取需要的信息,但也要关闭
#include <sys/types.h>
#include <dirent.h>
DIR *opendir(const char *name);
功能:该函数可以打开指定路径+名称的目录。
参数:name 目录所在的路径+名称
返回值:成功 目录流指针,指向一个打开的目录
失败 NULL
int closedir(DIR *dirp);
功能:关闭指定的目录流指针
参数:dirp 要关闭的目录
返回值:成功 0
失败 -1
遍历目录:
struct dirent *readdir(DIR *dirp);
功能:该函数可以在已经打开的目录中读取一次目录文件信息。
参数:dirp 已经打开的目录流指针
返回值:成功 返回文件信息结构体
失败/目录末尾 NULL
struct dirent {
ino_t d_ino; /* inode number /
off_t d_off; / not an offset; see NOTES /
unsigned short d_reclen; / length of this record /
unsigned char d_type; / type of file; not supported
by all filesystem types /
char d_name[256]; / filename */
};
// readdir的基本使用
DIR *tp =NULL;
tp =opendir (argv[1]);
struct dirent *tt ;
tt =readdir (tp);
if (tt==NULL)
{
return 0;
}
puts (tt->d_name);
}
closedir (tp);
用户时间的获取
我们可以发现,当我们使用sata函数来获取时间的时候,返回值是一个size_t类型的时间戳,当然我们并不能看懂这些时间,所以需要一个函数来对他们进行转化
头文件: time.h
函数: struct tm *localtime(const time_t *timep);
功能: 该函数可以将时间的秒值转换成年月日,时分秒格式。
参数: timep 时间秒的数字地址
返回值:成功 格式化后的结构体,tm
失败 NULL
时间结构体 tm 如下格式
struct tm {
int tm_sec; /* seconds */
int tm_min; /* minutes */
int tm_hour; /* hours */
int tm_mday; /* day of the month */
int tm_mon; /* month */
int tm_year; /* year */
3.2 实时获取当前时间
time_t time(time_t *t);
功能:该函数可以获取当前函数执行瞬间的秒数。
参数:t 用于存储时间秒的外部变量
返回值:成功 返回的 == t
失败 -1;
用户信息的获取
1、头文件 pwd.h == /etc/passwd
函数: getpwuid()== struct passwd *getpwuid(uid_t uid)
功能:通过该函数可以将uid对应的用户信息以返回值
形式获取。
参数: uid 用户id
返回值:成功 获取用户信息
失败 NULL
struct passwd {
char *pw_name; /* username */