獲取檔案屬性—stat、lstat、fstat
一、函式原型
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
int stat(const char *path, struct stat *buf);
int fstat(int fd, struct stat *buf);
int lstat(const char *path, struct stat *buf);
二、引數
1、path :檔名或者目錄名
2、fd : 檔案描述符
3、 struct stat {
dev_t st_dev; /* ID of device containing file */ // 檔案的裝置編號
ino_t st_ino; /* inode number */ // 結點
mode_t st_mode; /* protection */ // 檔案的型別和存取的許可權
nlink_t st_nlink; /* number of hard links */ // 連到該檔案的硬連結數目,新建的檔案則硬連線數為 1
uid_t st_uid; /* user ID of owner */ // 使用者ID
gid_t st_gid; /* group ID of owner */ // 組ID
dev_t st_rdev; /* device ID (if special file) */ // 若此檔案為裝置檔案,則為其裝置的編號
off_t st_size; /* total size, in bytes */ // 檔案位元組數(檔案大小)
blksize_t st_blksize; /* blocksize for filesystem I/O */ // 塊大小
blkcnt_t st_blocks; /* number of 512B blocks allocated */ // 塊數
time_t st_atime; /* time of last access */ // 最後一次訪問時間
time_t st_mtime; /* time of last modification */ // 最後一次修改時間
time_t st_ctime; /* time of last status change */ // 最後一次改變時間
};
st_mode :該變數佔 2 byte,共16位
(1)、掩碼的使用: st_mode & 掩碼
(2)、其他人許可權( 0-2 bit )
(a)、S_IROTH 00004 讀許可權
(b)、S_IWOTH 00002 寫許可權 掩碼:S_IRWXO
(c)、S_IXOTH 00001 執行許可權
(3)、所屬組許可權(3-5bit)
(a)、S_IRWXG 00070 讀許可權
(b)、S_IRGRP 00040 寫許可權 掩碼:S_RWXG
(c)、S_IXGRP 00010 執行許可權
(4)、檔案所有者許可權(6-8bit)
(a)、S_IRUSR 00400 讀許可權
(b)、S_IWUSR 00200 寫許可權 掩碼:S_IRWXU
(c)、S_IXUSR 00100 執行許可權
(5)、檔案特權位(9-11bit)
(a)、 S_ISUID 0004000 設定使用者ID
(b)、 S_ISGID 0002000 設定組ID 檔案特權位很少用
(6)、檔案型別(12-15bit)
(a) 、S_IFSOCK 0140000 socket(套接字)
(b) 、S_IFLNK 0120000 symbolic link(符號連結--軟連線)
(c) 、S_IFREG 0100000 regular file(普通檔案)
(d)、 S_IFBLK 0060000 block device(塊裝置) 掩碼:S_IFMT
(e) 、S_IFDIR 0040000 directory(目錄)
(f) 、 S_IFCHR 0020000 character device(字元裝置)
(g)、 S_IFIFO 0010000 FIFO(管道)
三、返回值
以上三個獲取檔案屬性的函式 若成功,返回0;若失敗,返回 -1;
四、stat、lstat、fstat之間的區別
1、fstat 函式:系統呼叫的是一個 ”檔案描述符”,而另外兩個則直接接收“檔案路徑”。檔案描述符是我們用 open 系統呼叫後得到的,而檔案全路徑直接寫就可以了。
2、stat 函式與 lstat 函式的區別: 當一個檔案是符號連結時,lstat 函式返回的是該符號連結本身的資訊;而 stat 函式返回的是該連結指向檔案的資訊。
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<stdio.h>
#include<stdlib.h>
int main(int argc,char *argv[])
{
if( argc<2 )
{
perror("a.out ");
exit(1);
}
struct stat st;
int ret = lstat(argv[1],&st);
if( ret == -1)
{
perror("lstat");
exit(1);
}
int size = st.st_size;
printf("file size = %d\n",size);
return 0;
}
五、使用 stat() 函式實現一個簡單的 ls -l Shell 命令:
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<pwd.h> // 所有者資訊
#include<grp.h> // 所屬組資訊
#include<time.h>
#include<string.h>
#include<stdlib.h>
#include<stdio.h>
int main(int argc,char *argv[])
{
if( argc<2 )
{
perror("./a.out filename\n");
exit(1);
}
struct stat st;
int i;
for( i = 1; i<argc; i++)
{
int ret = stat(argv[i],&st); // 獲取檔案或者目錄的所有資訊儲存於 st 結構體中
if( ret == -1 )
{
perror("stat");
exit(1);
}
// 儲存檔案型別和訪問許可權
char perms[11] = {0};
// 判斷檔案型別
switch( st.st_mode & S_IFMT )
{
case S_IFSOCK: // 套接字檔案
perms[0] = 's';
break;
case S_IFLNK: // 軟連線檔案
perms[0] = 'l';
break;
case S_IFREG: // 普通檔案
perms[0] = '-';
break;
case S_IFBLK: // 塊裝置檔案
perms[0] = 'b';
break;
case S_IFDIR: // 目錄檔案
perms[0] = 'd';
break;
case S_IFCHR: // 字元裝置檔案
perms[0] = 'c';
break;
case S_IFIFO: // 管道檔案
perms[0] = 'p';
break;
default:
break;
}
// 判斷檔案的訪問許可權
// 檔案的所有者
perms[1] = (st.st_mode & S_IRUSR) ? 'r':'-';
perms[2] = (st.st_mode & S_IWUSR) ? 'w':'-';
perms[3] = (st.st_mode & S_IXUSR) ? 'x':'-';
// 檔案的所屬組
perms[4] = (st.st_mode & S_IRGRP) ? 'r':'-';
perms[5] = (st.st_mode & S_IWGRP) ? 'w':'-';
perms[6] = (st.st_mode & S_IXGRP) ? 'x':'-';
// 檔案的其他使用者
perms[7] = (st.st_mode & S_IROTH) ? 'r':'-';
perms[8] = (st.st_mode & S_IWOTH) ? 'w':'-';
perms[9] = (st.st_mode & S_IXOTH) ? 'x':'-';
// 硬連結計數
int nums = st.st_nlink;
// 檔案所有者
char *fileuser = getpwuid(st.st_uid)->pw_name;
// 檔案所屬組
char *filegroup = getgrgid(st.st_gid)->gr_name;
// 檔案大小
int size = (int)st.st_size;
// 檔案修改時間
char *time = ctime(&st.st_mtime);
char mtime[512]="";
strncpy(mtime,time,strlen(time)-1);
// 儲存輸出資訊格式
char buf[1024]={0};
// 把對應資訊按格式輸出到 buf 中
sprintf(buf,"%s %d %s %s %d %s %s",perms,nums,fileuser,filegroup,size,mtime,argv[i]);
// 列印 buf
printf("%s\n",buf);
// drwxrwxr-x 3 arrayli arrayli 4096 11月 13 23:19 day05
// -rw-r--r-- 1 arrayli arrayli 8980 11月 7 22:05 examples.desktop
}
return 0;
}