欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  科技

Linux stat函数和stat命令

程序员文章站 2022-07-02 12:00:08
stat函数和stat命令 linux文件里的【inode = index node】解释:要理解inode必须了解磁盘和【目录项】,inode实际是连接【目录项】和磁盘的中间物质。 图里的大圈代表硬件的磁盘,里面的小圈代表某个文件存储在磁盘上了。 【inode = index node】的node ......

stat函数和stat命令

linux文件里的【inode = index node】解释:要理解inode必须了解磁盘和【目录项】,inode实际是连接【目录项】和磁盘的中间物质。

  • 图里的大圈代表硬件的磁盘,里面的小圈代表某个文件存储在磁盘上了。
  • 【inode = index node】的node(承载node信息的结构体是:stat,stat的定义在后面 )里面有:
    • 文件大小
    • 文件的最后修改时间
    • 文件的所属用户
    • 文件的权限
    • 硬链接计数(ls -l 显示出来的数字)
    • 块位置:指定文件存储在磁盘的具体位置。
  • 下图中的hello是个普通文件,hello.hard是hello的硬链接
  • 文件夹里放的就是每个文件的【目录项】如下图,【目录项】里有:
    • 文件名
    • 该目录项的大小
    • 文件的类型
    • inode

Linux stat函数和stat命令

  • 如何查看文件的【inode】呢?使用【-i】选项

    ls -li 文件名

    执行结果:

    ys@ys-virtualbox:~/lianxi1$ ls -li hello hello.hard 
    3801352 -rw-rw-r-- 2 ys ys 0 4月  24 11:01 hello
    3801352 -rw-rw-r-- 2 ys ys 0 4月  24 11:01 hello.hard

    发现hello和hello.hard的inode(3801352)是相同的,也就说明了,只在磁盘上存了一份。

  • 如何查看目录项呢?用emacs或者vim打开目录(lianxi1),截图如下。但是看不到文件的【inode】。

Linux stat函数和stat命令

1,stat函数:取得指定文件的文件属性,文件属性存储在结构体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 结构体:

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:设备id,不太常用

  • st_ino:【inode】,【inode】是啥?不知道就看上面关于【inode】的解释

  • st_mode:文件的类型和权限,共16位,如下图。

    • 0-11位控制文件的权限

    • 12-15位控制文件的类型

    0-2比特位:其他用户权限

    3-5比特位:组用户权限

    6-8比特位:本用户权限

    9-11比特位:特殊权限

    12-15比特位:文件类型(因为文件类型只有7中,所以用12-14位就够了

Linux stat函数和stat命令

文件类型的宏如下(下面的数字是8进制):

  • s_ifsock 0140000 socket
  • s_iflnk 0120000 symbolic link(软连接)
  • s_ifreg 0100000 regular file(普通文件)
  • s_ifblk 0060000 block device(块设备文件)
  • s_ifdir 0040000 directory(目录)
  • s_ifchr 0020000 character device(字符设备文件)
  • s_ififo 0010000 fifo(管道)
另一种文件类型的宏:       
       s_isreg(m)  is it a regular file?

       s_isdir(m)  directory?

       s_ischr(m)  character device?

       s_isblk(m)  block device?

       s_isfifo(m) fifo (named pipe)?

       s_islnk(m)  symbolic link?  (not in posix.1-1996.)

       s_issock(m) socket?  (not in posix.1-1996.)

文件权限的宏如下:

       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:硬连接计数

  • st_uid:这个文件所属用户的id

  • st_gid:这个文件所属用户的组id

  • st_rdev:特殊设备的id,不太常用

  • st_size:文件的大小

  • st_blksize:不明是干啥的

  • st_blocks:不明是干啥的

  • struct timespec st_atim:最后访问的时间

  • struct timespec st_mtim:最后修改的时间

  • struct timespec st_ctim:最后状态改变的时间

    struct timespec {
      __kernel_time_t tv_sec;  /* seconds */当前时间到1970.1.1 00:00:00的秒数
      long        tv_nsec;    /* nanoseconds *//纳秒数(不知道从哪到哪的)
    };
    1s  秒   = 1000ms 毫秒
    1ms 毫秒 = 1000us 微秒
    1us 微秒 = 1000ns 纳秒

pathname:文件名

返回值:0代表成功;-1代表失败,并设置error

例子:statbuf是结构体stat,可以看出来st_mode是个10进制的数字。

Linux stat函数和stat命令

  • st_mode

    用gdb显示st_mode,发现返回的st_mode是个10进制的数字,用gdb的【p/o】(o代表用8进制表示)命令把10进制的33204转换成了8进制的【0100664】,第一个0代笔是8进制,后三位的【100】代表文件类型,从上面的说明可以看出来【100】代表普通文件,最后三位的【664】代表这个文件的权限(本用户:rw-,组用户:rw-,其他用户:r--)。所以从st_mode里就可以得知文件的类型和权限设置(只使用了16个比特位,真的好节省空间,牛逼!)

  • st_uid

  • st_gid

    发现st_uid和st_gid是1000,但这个1000怎么和用户对应上呢,查看/etc/passwd文件,发现用于ys的uid和gid都是1000,所以就对应上了。

Linux stat函数和stat命令

stat命令,是stat函数对应,执行结果如下:

ys@ys-virtualbox:~/lianxi1$ stat hello
  file: hello
  size: 11          blocks: 8          io block: 4096   regular file
device: 801h/2049d  inode: 3801352     links: 2
access: (0764/-rwxrw-r--)  uid: ( 1000/      ys)   gid: ( 1000/      ys)
access: 2019-04-24 17:02:39.199461489 +0800
modify: 2019-04-24 16:54:16.407461489 +0800
change: 2019-04-24 17:03:44.927461489 +0800