linux 查看文件信息

这本阿里P8撰写的算法笔记,再次推荐给大家,身边不少朋友学完这本书最后加入大厂:

Github 疯传!史上最强悍!阿里大佬「LeetCode刷题手册」开放下载了!

利用系统调用stat()、lstat()以及fstat(),可获取与文件有关的信息,其中大部分提取自文件i节点。以上3个系统调用之间仅有的区别在于对文件的描述方式不同。

         1  stat()返回所命名文件的相关信息。

         2  lstat()与stat()类似,区别在于如果文件属于符号链接,那么所返回的信息针对的是符号链接自身(而非符号链接所指向的文件)。

         3  fstat()则返回由某个打开文件描述符所指代文件的相关信息。

         系统调用stat()和lstat()不需要对所操作的文件本身拥有任何权限,但针对指定pathname的父目录要有执行(搜索)权限。而只要为它提供有效的文件描述符,fstat()系统调用总是成功的。

         上述所有系统调用都会在缓冲区中返回一个由statbuf指向的stat结构,其格式如下:

         

设备ID和i节点号

         st_dev字段标识文件所驻留的设备。st_ino字段包含了文件的i节点号。利用这两个字段,可以在所有文件系统中唯一表示某个文件。

         如果是针对设备的i节点,那么st_rdev字段则包含设备的主、辅ID。利用宏major()和minor(),可以提取主、辅ID。在Linux系统上,要使用这两个宏,需要定义_BSD_SOURCE宏,然后include定义这两个宏的头文件<sys/types.h>。

         

         由于major()和minor()所返回的整型值大小随UINX实现的不同而各有不用。为保证可移植性,打印时总是将返回值强制转换为long。

 

文件所有权

         st_uid和st_gid字段分别表示文件的属主(用户ID)和属组(组ID)。

 

链接数

         st_nlink字段包含了指向文件的(硬)链接数。

 

文件类型及权限

         st_mode字段内含有位掩码,起表示文件类型和指定文件权限的双重作用。下图为该字段所含各位的布局情况。 

文件类型

权限

 

 

 

 

U

G

T

R

W

X

R

W

X

R

W

X

         Linux使用了st_mode字段中的4位来标识文件类型位。st_mode字段与常量S_IFMT相与(&),可从该字段中提取出文件类型。将计算结果与一系列常量进行比较,即可确定文件类型:

         if((statbuf.st_mode & S_IFMT) == S_IFREG)

                   printf(“regularfile\n”);

         在Linux中可利用标准宏将其简化:

         if(S_ISREG(statbuf.st_mode))

                   printf(“regularfile\n”);

 

常  量

测  试  宏

文  件  类  型

S_IFREG

S_ISREG()

常规文件

S_IFDIR

S_ISDIR()

目录

S_IFCHR

S_ISCHR()

字符设备

S_IFBLK

S_ISBLK()

块设备

S_IFIFO

S_ISFIFO()

FIFO或管道

S_IFSOCK

S_ISSOCK()

套接字

S_IFLNK

S_ISLNK()

符号链接

针对stat结构中的st_mode来检查文件类型的宏

         想从<sys/stat.h>中获取S_IFSOCK和S_ISSOCK()的定义,必须定义_BSD_SOURCE特性测试宏,或将_XOPEN_SOURCE定义为不小于500的值。

         st_mode字段的低12位定义了文件权限。最低9位分别用来表示文件属主、属组以及其他用户的读、写、执行权限。

 

文件大小、已分配块以及最优I/O块大小

         对于常规文件,st_size字段表示文件的字节数。对于符号链接,st_size字段则表示链接所指路径名的长度,以字节为单位。对于共享内存对象,该字段则表示对象的大小。

         st_blocks字段表示分配给文件的总块数,块大小为512字节,其中包括了为指针块所分配的空间。st_blocks字段记录了实际分配给文件的磁盘块数量。如果文件内含空洞,该值将小于从相应文件字节数字段(st_size)的值。

         st_blksize字段所指并非底层文件系统的块大小,而是针对文件系统上文件进行I/O操作时的最优块大小(以字节为单位)。一般而言,st_blksize的返回值为4096。

 

文件时间戳

         st_time、st_mtime和st_ctime字段,分别记录了对文件的上次访问时间、上次修改时间以及文件状态发送改变的上次时间。这3个字段的类型都是time_t,记录了自新纪元(Epoch)以来的秒数。

 

示例程序

(获取并解释文件的stat信息)

#define _BSD_SOURCE #include <sys/stat.h> #include <sys/types.h> #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <time.h> #include <string.h>

#define FP_SPECIAL 1 #define STR_SIZE sizeof("rwxrwxrwx")

typedef enum { FALSE, TRUE } Boolean;

static char *filePermStr(mode_t perm, int flags) { static char str[STR_SIZE]; snprintf(str, STR_SIZE, "%c%c%c%c%c%c%c%c%c", (perm & S_IRUSR) ? 'r' : '-', (perm & S_IWUSR) ? 'w' : '-', (perm & S_IXUSR) ? (((perm & S_ISUID) && (flags & FP_SPECIAL)) ? 's' : 'x') : (((perm & S_ISUID) && (flags & FP_SPECIAL)) ? 'S' : '-'), (perm & S_IRGRP) ? 'r' : '-', (perm & S_IWGRP) ? 'w' : '-', (perm & S_IXGRP) ? (((perm & S_ISGID) && (flags & FP_SPECIAL)) ? 's' : 'x') : (((perm & S_ISGID) && (flags & FP_SPECIAL)) ? 'S' : '-'), (perm & S_IROTH) ? 'r' : '-', (perm & S_IWOTH) ? 'w' : '-', (perm & S_IXOTH) ? (((perm & S_ISVTX) && (flags & FP_SPECIAL)) ? 't' : 'x') : (((perm & S_ISVTX) && (flags & FP_SPECIAL)) ? 'T' : '-'));

return str;

}

static void showStatInfo(struct stat *st) { printf("File type: ");

switch(st->st_mode & S_IFMT)
{
    case S_IFREG:    printf("regular file\n");        break;
    case S_IFDIR:    printf("directory\n");            break;
    case S_IFCHR:    printf("character device\n");    break;
    case S_IFBLK:    printf("block device\n");        break;
    case S_IFLNK:    printf("symbolic (soft) link\n");    break;
    case S_IFIFO:    printf("FIFO or pipe\n");        break;
    case S_IFSOCK:    printf("socket\n");                break;
    default:        printf("unknown file type?\n");    break;
}

printf("Device containing i-node:     major=%ld    minor=%ld\n", 
    (long) major(st->st_dev), (long) minor(st->st_dev));

printf("I-node number:            %ld\n", (long) st->st_ino);
printf("Mode:                %lo (%s)\n", (unsigned long) st->st_mode, 
    filePermStr(st->st_mode, 0));

if (st->st_mode & (S_ISUID| S_ISGID| S_ISVTX))
    printf("    special bits set:    %s%s%s\n",
        (st->st_mode & S_ISUID) ? "set-UID " : "",
        (st->st_mode & S_ISGID) ? "set-GID " : "",
        (st->st_mode & S_ISVTX) ? "sticky " : "");

printf("Number of (hard) links:        %ld\n", (long) st->st_nlink);
printf("Ownership:            UID=%ld    GID=%ld\n",
    (long) st->st_uid, (long) st->st_gid);

if (S_ISCHR(st->st_mode) || S_ISBLK(st->st_mode))
    printf("Device number (st_rdev):        major=%ld; minor=%ld\n",
        (long) major(st->st_rdev), (long) minor(st->st_rdev));

printf("File size:            %lld bytes\n", (long long) st->st_size);
printf("Optimal I/O block size:        %ld bytes\n", (long) st->st_blksize);
printf("512B blocks allocated:        %lld\n", (long long) st->st_blocks);
printf("Last file access:        %s\n", ctime(&st->st_atime));
printf("Last file modification:        %s\n", ctime(&st->st_mtime));
printf("Last status change:        %s\n", ctime(&st->st_ctime));

}

int main(int argc, char *argv[]) { struct stat st; Boolean statLink; int fname;

statLink = (argc > 1) && strcmp(argv[1], "-l") == 0;

fname = statLink ? 2: 1;

if (fname >= argc || (argc > 1 && strcmp(argv[1], "--help") == 0))
    printf("%s [-l] file\n         -l = use lstat() instead of stat()\n", argv[0]);

if (statLink)
{
    if (lstat(argv[fname], &st) == -1)
    {
        printf("lstat() operate failed!\n");
    }
}
else
{
    if (stat(argv[fname], &st) == -1)
        printf("stat() operate failed!\n");
}    
showStatInfo(&st);
exit(1);

}

以下是运行结果:

———————————————— 版权声明:本文为优快云博主「davidsky11」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 原文链接:https://blog.youkuaiyun.com/davidsky11/article/details/25425009

这本阿里P8撰写的算法笔记,再次推荐给大家,身边不少朋友学完这本书最后加入大厂:

Github 疯传!史上最强悍!阿里大佬「LeetCode刷题手册」开放下载了!

以上就是良许教程网为各位朋友分享的Linux相关知识。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值