Linux中stat函數(shù)和stat命令使用詳解
stat函數(shù)和stat命令
linux文件里的【inode = index node】解釋?zhuān)阂斫鈏node必須了解磁盤(pán)和【目錄項(xiàng)】,inode實(shí)際是連接【目錄項(xiàng)】和磁盤(pán)的中間物質(zhì)。
圖里的大圈代表硬件的磁盤(pán),里面的小圈代表某個(gè)文件存儲(chǔ)在磁盤(pán)上了。
【inode = index node】的node(承載node信息的結(jié)構(gòu)體是:stat,stat的定義在后面 )里面有:
文件大小 文件的最后修改時(shí)間 文件的所屬用戶 文件的權(quán)限 硬鏈接計(jì)數(shù)(ls -l 顯示出來(lái)的數(shù)字) 塊位置:指定文件存儲(chǔ)在磁盤(pán)的具體位置。下圖中的hello是個(gè)普通文件,hello.hard是hello的硬鏈接
文件夾里放的就是每個(gè)文件的【目錄項(xiàng)】如下圖,【目錄項(xiàng)】里有:
文件名 該目錄項(xiàng)的大小 文件的類(lèi)型 inode如何查看文件的【inode】呢?使用【-i】選項(xiàng)
ls -li 文件名
執(zhí)行結(jié)果:
ys@ys-VirtualBox:~/lianxi1$ ls -li hello hello.hard 3801352 -rw-rw-r-- 2 ys ys 0 4月 24 11:01 hello3801352 -rw-rw-r-- 2 ys ys 0 4月 24 11:01 hello.hard
發(fā)現(xiàn)hello和hello.hard的inode(3801352)是相同的,也就說(shuō)明了,只在磁盤(pán)上存了一份。
如何查看目錄項(xiàng)呢?用emacs或者vim打開(kāi)目錄(lianxi1),截圖如下。但是看不到文件的【inode】。
1,stat函數(shù):取得指定文件的文件屬性,文件屬性存儲(chǔ)在結(jié)構(gòu)體stat里。
#include <sys/types.h>#include <sys/stat.h>#include <unistd.h>int stat(const char *pathname, struct stat *statbuf);int fstat(int fd, struct stat *statbuf);int lstat(const char *pathname, struct stat *statbuf);
struct stat 結(jié)構(gòu)體:
struct stat { dev_t st_dev; /* ID of device containing file */ ino_t st_ino; /* Inode number */ mode_t st_mode; /* File type and mode */ nlink_t st_nlink; /* Number of hard links */ uid_t st_uid; /* User ID of owner */ gid_t st_gid; /* Group ID of owner */ dev_t st_rdev; /* Device ID (if special file) */ off_t st_size; /* Total size, in bytes */ blksize_t st_blksize; /* Block size for filesystem I/O */ blkcnt_t st_blocks; /* Number of 512B blocks allocated */ /* Since Linux 2.6, the kernel supports nanosecond precision for the following timestamp fields. For the details before Linux 2.6, see NOTES. */ struct timespec st_atim; /* Time of last access */ struct timespec st_mtim; /* Time of last modification */ struct timespec st_ctim; /* Time of last status change */ #define st_atime st_atim.tv_sec /* Backward compatibility */ #define st_mtime st_mtim.tv_sec #define st_ctime st_ctim.tv_sec };
st_dev:設(shè)備ID,不太常用st_ino:【inode】,【inode】是啥?不知道就看上面關(guān)于【inode】的解釋st_mode:文件的類(lèi)型和權(quán)限,共16位,如下圖。
0-11位控制文件的權(quán)限 12-15位控制文件的類(lèi)型0-2比特位:其他用戶權(quán)限3-5比特位:組用戶權(quán)限6-8比特位:本用戶權(quán)限9-11比特位:特殊權(quán)限12-15比特位:文件類(lèi)型(因?yàn)槲募?lèi)型只有7中,所以用12-14位就夠了
文件類(lèi)型的宏如下(下面的數(shù)字是8進(jìn)制):
S_IFSOCK 0140000 socket S_IFLNK 0120000 symbolic link(軟連接) S_IFREG 0100000 regular file(普通文件) S_IFBLK 0060000 block device(塊設(shè)備文件) S_IFDIR 0040000 directory(目錄) S_IFCHR 0020000 character device(字符設(shè)備文件) S_IFIFO 0010000 FIFO(管道)判斷文件類(lèi)型的函數(shù),返回true,false S_ISREG(stat.st_mode) is it a regular file? S_ISDIR(stat.st_mode) directory? S_ISCHR(stat.st_mode) character device? S_ISBLK(stat.st_mode) block device? S_ISFIFO(m) FIFO (named pipe)? S_ISLNK(stat.st_mode) symbolic link? (Not in POSIX.1-1996.) S_ISSOCK(stat.st_mode) socket? (Not in POSIX.1-1996.)
文件權(quán)限的宏如下:
S_ISUID 04000 set-user-ID bit S_ISGID 02000 set-group-ID bit (see below) S_ISVTX 01000 sticky bit (see below) S_IRWXU 00700 owner has read, write, and execute permission S_IRUSR 00400 owner has read permission S_IWUSR 00200 owner has write permission S_IXUSR 00100 owner has execute permission S_IRWXG 00070 group has read, write, and execute permission S_IRGRP 00040 group has read permission S_IWGRP 00020 group has write permission S_IXGRP 00010 group has execute permission S_IRWXO 00007 others (not in group) have read, write, and execute permission S_IROTH 00004 others have read permission S_IWOTH 00002 others have write permission S_IXOTH 00001 others have execute permission st_nlink:硬連接計(jì)數(shù) st_uid:這個(gè)文件所屬用戶的ID st_gid:這個(gè)文件所屬用戶的組ID st_rdev:特殊設(shè)備的ID,不太常用 st_size:文件的大小 st_blksize:不明是干啥的 st_blocks:不明是干啥的 struct timespec st_atim:最后訪問(wèn)的時(shí)間 struct timespec st_mtim:最后修改的時(shí)間 struct timespec st_ctim:最后狀態(tài)改變的時(shí)間
struct timespec {__kernel_time_ttv_sec; /* seconds */當(dāng)前時(shí)間到1970.1.1 00:00:00的秒數(shù)longtv_nsec;/* nanoseconds *//納秒數(shù)(不知道從哪到哪的)};1s 秒 = 1000ms 毫秒1ms 毫秒 = 1000us 微秒1us 微秒 = 1000ns 納秒
pathname:文件名
返回值:0代表成功;-1代表失敗,并設(shè)置error
例子:statbuf是結(jié)構(gòu)體stat,可以看出來(lái)st_mode是個(gè)10進(jìn)制的數(shù)字。
st_mode
用gdb顯示st_mode,發(fā)現(xiàn)返回的st_mode是個(gè)10進(jìn)制的數(shù)字,用gdb的【p/o】(o代表用8進(jìn)制表示)命令把10進(jìn)制的33204轉(zhuǎn)換成了8進(jìn)制的【0100664】,第一個(gè)0代筆是8進(jìn)制,后三位的【100】代表文件類(lèi)型,從上面的說(shuō)明可以看出來(lái)【100】代表普通文件,最后三位的【664】代表這個(gè)文件的權(quán)限(本用戶:rw-,組用戶:rw-,其他用戶:r--)。所以從st_mode里就可以得知文件的類(lèi)型和權(quán)限設(shè)置(只使用了16個(gè)比特位,真的好節(jié)省空間,牛逼!)
st_uid
st_gid
發(fā)現(xiàn)st_uid和st_gid是1000,但這個(gè)1000怎么和用戶對(duì)應(yīng)上呢,查看/etc/passwd文件,發(fā)現(xiàn)用于ys的uid和gid都是1000,所以就對(duì)應(yīng)上了。
stat命令,是stat函數(shù)對(duì)應(yīng),執(zhí)行結(jié)果如下:
ys@ys-VirtualBox:~/lianxi1$ stat hello File: hello Size: 11 Blocks: 8 IO Block: 4096 regular fileDevice: 801h/2049dInode: 3801352 Links: 2Access: (0764/-rwxrw-r--) Uid: ( 1000/ ys) Gid: ( 1000/ ys)Access: 2019-04-24 17:02:39.199461489 +0800Modify: 2019-04-24 16:54:16.407461489 +0800Change: 2019-04-24 17:03:44.927461489 +0800
2,getpwuid函數(shù):返回/etc/passwd文件里指定uid的行,把這一行的信息放入結(jié)構(gòu)體passwd中。雖然返回值是指針,但不需要調(diào)用free函數(shù)。
#include <sys/types.h>#include <pwd.h>struct passwd *getpwnam(const char *name);struct passwd *getpwuid(uid_t uid);struct passwd { char *pw_name; /* username */ char *pw_passwd; /* user password */ uid_t pw_uid; /* user ID */ gid_t pw_gid; /* group ID */ char *pw_gecos; /* user information */ char *pw_dir; /* home directory */ char *pw_shell; /* shell program */};
3,getgrgid函數(shù):返回/etc/group文件里指定gid的行,把這一行的信息放入結(jié)構(gòu)體group中。雖然返回值是指針,但不需要調(diào)用free函數(shù)。
#include <sys/types.h>#include <grp.h>struct group *getgrnam(const char *name);struct group *getgrgid(gid_t gid);struct group { char *gr_name; /* group name */ char *gr_passwd; /* group password */ gid_t gr_gid; /* group ID */ char **gr_mem; /* NULL-terminated array of pointersto names of group members */};
4,localtime函數(shù):傳入從stat函數(shù)里得到的st_mtim.tv_sec(當(dāng)前時(shí)間到1970.1.1 00:00:00的秒數(shù)),得到結(jié)構(gòu)體tm。雖然返回值是指針,但不需要調(diào)用free函數(shù)。
#include <time.h>struct tm *localtime(const time_t *timep);struct tm { int tm_sec; /* Seconds (0-60) */ int tm_min; /* Minutes (0-59) */ int tm_hour; /* Hours (0-23) */ int tm_mday; /* Day of the month (1-31) */ int tm_mon; /* Month (0-11) */ int tm_year; /* Year - 1900 */ int tm_wday; /* Day of the week (0-6, Sunday = 0) */ int tm_yday; /* Day in the year (0-365, 1 Jan = 0) */ int tm_isdst; /* Daylight saving time */};
5,lstat函數(shù):stat碰到軟鏈接,會(huì)追述到源文件,穿透;lstat并不會(huì)穿透。
例子:模仿ls -l 文件
#include <sys/types.h>#include <sys/stat.h>#include <unistd.h>#include <stdio.h>#include <string.h>#include <pwd.h>//getpwuid#include <stdlib.h>#include <time.h>//localtime#include <grp.h>//getgrgidint main(int argc, char* argv[]){ struct stat sbuf; //stat(argv[1], &sbuf); lstat(argv[1], &sbuf); char str[11] = {0}; memset(str, ’-’, (sizeof str - 1)); //文件類(lèi)型 if(S_ISREG(sbuf.st_mode)) str[0] = ’-’; if(S_ISDIR(sbuf.st_mode)) str[0] = ’d’; if(S_ISCHR(sbuf.st_mode)) str[0] = ’c’; if(S_ISBLK(sbuf.st_mode)) str[0] = ’b’; if(S_ISFIFO(sbuf.st_mode)) str[0] = ’p’; if(S_ISLNK(sbuf.st_mode)) str[0] = ’l’; if(S_ISSOCK(sbuf.st_mode)) str[0] = ’s’; //本用戶的文件權(quán)限 if(sbuf.st_mode & S_IRUSR) str[1] = ’r’; if(sbuf.st_mode & S_IWUSR) str[2] = ’w’; if(sbuf.st_mode & S_IXUSR) str[3] = ’x’; //本用戶的組的文件權(quán)限 if(sbuf.st_mode & S_IRGRP) str[4] = ’r’; if(sbuf.st_mode & S_IWGRP) str[5] = ’w’; if(sbuf.st_mode & S_IXGRP) str[6] = ’x’; //其他用戶的文件權(quán)限 if(sbuf.st_mode & S_IROTH) str[7] = ’r’; if(sbuf.st_mode & S_IWOTH) str[8] = ’w’; if(sbuf.st_mode & S_IXOTH) str[9] = ’x’; char ymd[20] = {0}; //取得日期和時(shí)間 struct tm* tm = localtime(&sbuf.st_atim.tv_sec); sprintf(ymd, '%2d月 %2d %02d:%02d', tm->tm_mon + 1, tm->tm_mday, tm->tm_hour + 1,tm->tm_sec); //-rw-r--r-- 1 ys ys 134 4月 25 09:21 st2.c printf('%s %ld %s %s %ld %s %sn', str, sbuf.st_nlink, getpwuid(sbuf.st_uid)->pw_name, getgrgid(sbuf.st_gid)->gr_name, sbuf.st_size, ymd, argv[1]); return 0;}
6,access函數(shù):判斷調(diào)用程序的用戶對(duì)于指定文件的權(quán)限(可讀?可寫(xiě)?可執(zhí)行?)
#include <unistd.h>int access(const char *pathname, int mode);
例子:
#include <stdio.h>#include <unistd.h>//accessint main(int argc, char* argv[]){ if(access(argv[1], R_OK) == 0) printf('read okn'); if(access(argv[1], W_OK) == 0) printf('write okn'); if(access(argv[1], X_OK) == 0) printf('exe okn'); if(access(argv[1], F_OK) == 0) printf('existsn');}
先用ls -l 查看/usr/include/time.h文件的權(quán)限,結(jié)果如下
ys@ys-VirtualBox:~/lianxi$ ls -l /usr/include/time.h-rw-r--r-- 1 root root 10360 4月 17 2018 /usr/include/time.h
用ys用戶執(zhí)行例子程序,查看/usr/include/time.h文件,結(jié)果如下。因?yàn)閠ime.h是屬于root用戶的,對(duì)于其他用戶來(lái)說(shuō)是[r--],所以得出下面的結(jié)果。
ys@ys-VirtualBox:~/lianxi$ ./ac /usr/include/time.hread okexists
還是用ys用戶執(zhí)行,但是加上sudo,結(jié)果如下。發(fā)現(xiàn)結(jié)果和root用戶相同。因?yàn)榧恿藄udo,就編程了root用戶。
ys@ys-VirtualBox:~/lianxi$ sudo ./ac /usr/include/time.h[sudo] password for ys: read okwrite okexists
7,truncate函數(shù):截?cái)辔募蛿U(kuò)展文件的大小
#include <unistd.h>#include <sys/types.h>int truncate(const char *path, off_t length);
path:文件length:length大于原來(lái)文件的大小,則擴(kuò)展文件的大小至lengthlength小于原來(lái)文件的大小,則截?cái)辔募拇笮≈羖ength
8,link函數(shù):創(chuàng)建硬鏈接
#include <unistd.h>int link(const char *oldpath, const char *newpath);
返回值:成功返回0,失敗返回-1,并設(shè)置errno。
9,symlink函數(shù):創(chuàng)建軟鏈接
#include <unistd.h>int symlink(const char *target, const char *linkpath);
返回值:成功返回0,失敗返回-1,并設(shè)置errno。
10,readlink函數(shù):找到軟鏈接對(duì)應(yīng)的實(shí)際文件,把文件的名字放入buf里。注意:硬鏈接不行。
#include <unistd.h>ssize_t readlink(const char *pathname, char *buf, size_t bufsiz);
返回值:成功返回寫(xiě)入buf的字節(jié)數(shù),失敗返回-1,并設(shè)置errno。
11,unlink函數(shù):刪除軟硬鏈接,也可以刪除文件。
#include <unistd.h>int unlink(const char *pathname);
返回值:成功返回0,失敗返回-1,并設(shè)置errno。
有個(gè)特殊用法:下面的open代碼想要?jiǎng)?chuàng)建hello文件,然后直接用unlink刪除,但是能寫(xiě)入成功,ret是大于0的,程序執(zhí)行完,發(fā)現(xiàn)沒(méi)有做成hello文件。
結(jié)論:當(dāng)執(zhí)行unlink后,計(jì)數(shù)為0后,但,發(fā)現(xiàn)別的進(jìn)程還引用這個(gè)文件,這個(gè)時(shí)間點(diǎn),unlink不會(huì)刪除這個(gè)文件,等這個(gè)進(jìn)程結(jié)束后,再刪除,所以下面的write代碼能夠?qū)懭氤晒Α@眠@個(gè)特點(diǎn)可以實(shí)現(xiàn):在線觀看視頻時(shí),實(shí)際是把視頻文件下載到了本地(然后代碼里,使用unlink),看完后視頻文件的計(jì)數(shù)為0,就自動(dòng)刪除了,不怕視頻被泄露出去。
#include <unistd.h>#include <sys/types.h>#include <stdio.h>#include <sys/stat.h>#include <fcntl.h>int main(){ int fd = open('hello', O_WRONLY | O_CREAT, 0666); unlink('hello'); int ret = write(fd, 'aaa', 4); if(ret > 0){ printf('write OKn'); } }
12,chown函數(shù):改變文件的所屬用戶和組
#include <unistd.h>int chown(const char *pathname, uid_t owner, gid_t group);
pathname:文件
owner:用戶ID(數(shù)字的)/etc/passwd
group:組ID(數(shù)字的)/etc/group
返回值:0成功,-1失敗。
13,rename函數(shù):重命名
#include <stdio.h>int rename(const char *oldpath, const char *newpath);
oldpath :原來(lái)的文件名后者目錄
newpath:新的文件名后者目錄
返回值:0成功,-1失敗。
14,getcwd函數(shù):獲得當(dāng)前工作的目錄
#include <unistd.h>char *getcwd(char *buf, size_t size);
buf:當(dāng)前工作的目錄
size:緩沖區(qū)大小
返回值:成功返回當(dāng)前工作的目錄 失敗返回NULL
15,chdir函數(shù):改變進(jìn)程的工作目錄
#include <unistd.h>int chdir(const char *path);
path:目標(biāo)工作目錄
返回值:0成功,-1失敗
16,mkdir函數(shù):創(chuàng)建目錄
#include <sys/stat.h>#include <sys/types.h>int mkdir(const char *pathname, mode_t mode);
pathname:目標(biāo)工作目錄mode:mode & ~umask & 0777 。注意,如果沒(méi)有x權(quán)限,則無(wú)法cd進(jìn)入這個(gè)目錄。返回值:0成功,-1失敗
17,rmdir函數(shù):刪除目錄,目錄必須是空目錄,也就是里面沒(méi)有任何文件。
#include <unistd.h>int rmdir(const char *pathname);
18,opendir函數(shù):打開(kāi)目錄
#include <sys/types.h>#include <dirent.h>DIR *opendir(const char *name);
name:目錄名
返回值:a pointer to the directory stream
19,readdir函數(shù):讀目錄
#include <sys/types.h>#include <dirent.h>DIR *opendir(const char *name);
dirp:opendir函數(shù)的返回值
返回值:結(jié)構(gòu)體dirent,可以理解成最上面說(shuō)的【目錄項(xiàng)】NULL代表讀到末尾或者有錯(cuò)誤 NULL以外代表目錄項(xiàng)的內(nèi)容
20,closedir函數(shù):關(guān)閉目錄
#include <sys/types.h>#include <dirent.h>int closedir(DIR *dirp);
dirp:opendir函數(shù)的返回值
21,strerron函數(shù):打印出errno對(duì)應(yīng)的文字信息。
#include <string.h>char *strerror(int errnum);
errnum的宏放在文件:/usr/include/asm-generic/errno.h
例子:
#include <string.h>#include <stdio.h>#include <asm-generic/errno.h>//EDEADLKint main(){ char* buf = strerror(EDEADLK); printf('%sn', buf);//Resource deadlock avoided}
22,dup和dup2函數(shù):文件描述符的重定向
#include <unistd.h>int dup(int oldfd);int dup2(int oldfd, int newfd);
dup:和open類(lèi)似,先打開(kāi)一個(gè)新的文件描述符,讓新的文件描述符也指向:oldfd指向的地方。成功返回新打開(kāi)的文件描述符;失敗返回-1.
dup2:先消除newfd的指向再讓newfd指向oldfd指向的地方成功返回newfd;失敗返回-1.
例子:調(diào)用printf2次,第一次printf把內(nèi)容寫(xiě)到文件;第二次printf把內(nèi)容打印到屏幕。
#include <stdio.h>#include <unistd.h>#include <sys/types.h>#include <sys/stat.h>#include <fcntl.h>int main(){ int oldfd = dup(STDOUT_FILENO); int fd = open('www', O_WRONLY | O_CREAT, 0666); dup2(fd, STDOUT_FILENO); printf('aaaan'); fflush(stdout); int ret = dup2(oldfd, STDOUT_FILENO); //int ret = dup2(oldfd, 6); //perror('dup2:'); printf('reg:%dn', ret); printf('aaaan'); close(fd);}
到此這篇關(guān)于Linux中stat函數(shù)和stat命令使用詳解的文章就介紹到這了,更多相關(guān)Linux stat函數(shù)和stat命令內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!
相關(guān)文章:
1. Win10專(zhuān)業(yè)版激活錯(cuò)誤代碼0x803f7001要怎么解決?2. CentOS上的安全防護(hù)軟件Selinux詳解3. 電腦聲音太小如何增強(qiáng)?Win8增強(qiáng)電腦聲音的操作技巧4. Win8系統(tǒng)有雜音怎么處理?Win8系統(tǒng)有雜音處理辦法5. Kylin(麒麟系統(tǒng)) 同時(shí)掛載USB閃存盤(pán)和移動(dòng)硬盤(pán)的方法及注意事項(xiàng)6. UOS應(yīng)用商店不能用怎么辦? uos系統(tǒng)應(yīng)用商店不能使用的解決辦法7. OPENBSD上的ADSL和防火墻設(shè)置配置8. deepin20系統(tǒng)字體怎么設(shè)置? deepin終端字體大小的設(shè)置方法9. Debian圖標(biāo)怎么橫向排列? Debian11 Xfce橫排桌面圖標(biāo)的技巧10. Win8系統(tǒng)提示沒(méi)有權(quán)限使用相機(jī)的解決方法
