牛骨文教育服务平台(让学习变的简单)
博文笔记

Linux stat函数获取文件属性(文件大小,创建时间等,判断普通文件或者目录等)

创建时间:2017-01-01 投稿人: 浏览次数:3731

(注意:

1.部分平台不支持stat64,


2.获取文件大小的类型 off_t,


在linux中off_t类型默认是32位的long int(4个字节),


文件太大时可能会溢出。


建议gcc编译时加上-D_FILE_OFFSET_BITS=64选项,


off_t将会是8个字节的类型。


3.获取文件大小,很多人会想到C语言的ftell函数,但该函数


局限性很大,特别是对于大文件。)


(1)函数的定义


表头文件:    #include <sys/stat.h>
                     #include <unistd.h>
定义函数:    int stat(const char *file_name, struct stat *buf);

函数说明:   通过文件名filename获取文件信息,并保存在buf所指的结构体stat中
返回值:     执行成功则返回0,失败返回-1,错误代码存于errno
错误代码:
    ENOENT         参数file_name指定的文件不存在
    ENOTDIR        路径中的目录存在但却非真正的目录
    ELOOP          欲打开的文件有过多符号连接问题,上限为16符号连接
    EFAULT         参数buf为无效指针,指向无法存在的内存空间
    EACCESS        存取文件时被拒绝
    ENOMEM         核心内存不足
    ENAMETOOLONG   参数file_name的路径名称太长



(2)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则定义了下列数种情况:
    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
    若一目录具有sticky位(S_ISVTX),则表示在此目录下的文件只能被该文件所有者、此目录所有者或root来删除或改名。

使用stat函数最多的可能是ls-l命令,用其可以获得有关一个文件的所有信息。



(3)其他相关函数



函数都是获取文件(普通文件,目录,管道,socket,字符,块()的属性。
函数原型
#include <sys/stat.h>
int stat(const char *restrict pathname, struct stat *restrict buf);
提供文件名字,获取文件对应属性。


int fstat(int filedes, struct stat *buf);
通过文件描述符获取文件对应的属性。


int lstat(const char *restrict pathname, struct stat *restrict buf);
提供文件名字,获取文件属性。


主要区别:


 fstat(int fd,struct stat *)接收的已open的文件描述符


stat(char *filename,struct stat *)接收的路径名, 需要注意的是 能处理符号链接,但处理的是符号链接指向的文件。

lstat(char *filename,struct stat *)接收的路径名  ,需要注意的是,也能能处理符号链接,但处理的是符号链接本身(自身)文件。 




#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <dirent.h>
#include <errno.h>
#include <sys/stat.h>

#include <sys/types.h>
#include <sys/stat.h>
#include <time.h>

#if 1
typedef struct stat Stat_t;
#define StatFunc(x,y) stat(x,y)
#else
typedef struct stat64 Stat_t;  //部分平台不支持
#define StatFunc(x,y) stat64(x,y)
#endif

int func(void)
{
   char *filename ="binxu.rmvb" ;
   int nRet = 0;
   Stat_t s;
   time_t tTimeTmp = 0;
   struct tm stuTimeTmp;
   memset(&s, 0, sizeof(Stat_t));
   nRet = StatFunc(filename, &s);
   if (nRet)
   {
       printf("stat %s failed! error_code: %s", filename ,strerror(errno));
       return -1;
   }

   printf("
 s.st_size=%d,s.st_ctime=%d,s.st_mtime=%d,s.st_atime=%d 
",
          s.st_size,s.st_ctime,s.st_mtime,s.st_atime);

   if(s.st_size >0 )
   {
		 printf("
 size ========%0.3lf KB  %0.3lf M   %0.3lf G ======
",
          s.st_size/1024.0,s.st_size/1024.0/1024.0,s.st_size/1024.0/1024.0/1024.0);
   }

   memset(&stuTimeTmp,0,sizeof(struct tm));
   tTimeTmp = (s.st_ctime);
   //struct tm *gmtime_r(const time_t *timep, struct tm *result);
   (void *)gmtime_r(&tTimeTmp,&stuTimeTmp);
 
	printf("
 create time** %d-%d-%d  %d:%d:%d **
",
		stuTimeTmp.tm_year + 1900,
		stuTimeTmp.tm_mon + 1,
		stuTimeTmp.tm_mday,
		stuTimeTmp.tm_hour,
		stuTimeTmp.tm_min,
		stuTimeTmp.tm_sec );
    
   return 0;
}
  
int main()  
{  
    func();
}









stat获取到的文件属性的时间值是UTC时间。


在windows下看文件属性,时间+8北京时区刚好。





(4)利用stat 的属性判断属于是否属于同一挂载目录下的文件夹或者是否是同一设备


#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <dirent.h>
#include <errno.h>
#include <sys/stat.h>

#include <sys/types.h>
#include <sys/stat.h>
#include <time.h>

#if 1
typedef struct stat Stat_t;
#define StatFunc(x,y) stat(x,y)
#else
typedef struct stat64 Stat_t;  //部分平台不支持
#define StatFunc(x,y) stat64(x,y)
#endif

int func(char *pchFilename)
{
   int nRet = 0;
   Stat_t stuStat;
   dev_t mountDevice;
   
   if(NULL == pchFilename || 0 == strlen(pchFilename))
   {
		return -1;
   }
   
   memset(&stuStat, 0, sizeof(Stat_t));
   nRet = StatFunc(pchFilename, &stuStat);
   if (nRet)
   {
       printf("stat %s failed! error_code: %s", pchFilename ,strerror(errno));
       return -1;
   }

	printf("

**************filepath[%s]***************	",pchFilename);

	printf("
 (stuStat.st_mode & S_IFMT) == S_IFBLK  result is %d  ",(stuStat.st_mode & S_IFMT) == S_IFBLK);
   
   if ((stuStat.st_mode & S_IFMT) == S_IFBLK)
	{
	    mountDevice = stuStat.st_rdev; // //(设备类型)若此文件为设备文件,则为其设备编号
	}
	else
	{
		mountDevice = stuStat.st_dev;     //文件的设备编号
	}

	printf("
 st_rdev[%llu]  st_dev[%llu] st_ino[%lu]
", stuStat.st_rdev,stuStat.st_dev,stuStat.st_ino );
	
   return 0;
}
  
int main()  
{  
    func("/share/111/");
	func("/share/111/222/");
	func("/");
	
	func("/dev/sda1");
	func("/dev/sda2");
	func("/dev/sda5");

}




属于设备的,如/dev/sda1,(stuStat.st_mode & S_IFMT) == S_IFBLK条件为真,此时比较是否为同一设备,应该比较st_rdev属性

属于文件夹的,(stuStat.st_mode & S_IFMT) == S_IFBLK条件为假,判断是否同一个挂载点时,应该比较st_dev属性。



声明:该文观点仅代表作者本人,牛骨文系教育信息发布平台,牛骨文仅提供信息存储空间服务。