stat系统调用系列包括了fstat、stat和lstat,它们都是用来返回“相关文件状态信息”的,三者的不同之处在于设定源文件的方式不同。首先介绍的是一个非常重要的结构体类型,名字叫做struct stat。可以说,没有这个struct stat的支持,上述三个系统调用将寸步难行。这个struct stat结构体在不同的UNIX/Linux系统中的定义是有小的区别的,但你完全不用担心,这并不会影响我们的使用。

struct stat结构体中我们常用的且各个平台都一定有的域是:
st_mode 文件权限和文件类型信息
st_ino   与该文件关联的inode
st_dev   保存文件的设备
st_uid   文件属主的UID号
st_gid   文件属主的GID号
st_atime 文件上一次被访问的时间
st_ctime 文件的权限、属主、组或内容上一次被修改的时间
st_mtime 文件的内容上一次被修改的时间。
st_nlink  该文件上硬连接的个数


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;     //最后一次改变时间(指属性)
};

可以分别提取solaris(UNIX)和fedora(Linux)的struct stat结构体的原始定义比对一下便可以发现两者确实有所不同,但主要的域是完全相同的。st_mode这个域不像其他域那么容易使用,其他的域的值显而易见,而st_mode域是需要一些宏予以配合才能使用的。其实,通俗说,这些宏就是一些特定位置为1的二进制数的外号,我们使用它们和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:文件设置了SUID位
S_ISGID:文件设置了SGID位
S_ISVTX:文件设置了sticky位
用于解释st_mode标志的掩码包括:

   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             其他用户具可执行权限
S_IFMT:文件类型
S_IRWXU:属主的读/写/执行权限,可以分成S_IXUSR, S_IRUSR, S_IWUSR
S_IRWXG:属组的读/写/执行权限,可以分成S_IXGRP, S_IRGRP, S_IWGRP
S_IRWXO:其他用户的读/写/执行权限,可以分为S_IXOTH, S_IROTH, S_IWOTH



还有一些用于帮助确定文件类型的宏定义,这些和上面的宏不一样,这些是带有参数的宏,类似与函数的使用方法:
S_ISBLK (st_mode) :测试是否是特殊的块设备文件
S_ISCHR (st_mode) :测试是否是特殊的字符设备文件
S_ISDIR (st_mode) :测试是否是目录
S_ISFIFO (st_mode):测试是否是FIFO设备
S_ISREG (st_mode):测试是否是普通文件
S_ISLNK (st_mode) :测试是否是符号链接
S_ISSOCK (st_mode):测试是否是socket


原型是:
int fstat(int filedes, struct stat *buf);
int stat(const char *path, struct stat *buf);
int lstat(const char *path, struct stat *buf);

fstat区别于另外两个系统调用的地方在于,fstat系统调用接受的是 一个“文件描述符”,而另外两个则直接接受“文件全路径”。文件描述符是需要我们用open系统调用后才能得到的,而文件全路经直接写就可以了。
stat和lstat的区别:当文件是一个符号链接时,lstat返回的是该符号链接本身的信息;而stat返回的是该链接指向的文件的信息.



stat 和lstat的区别:当文件是一个符号链接时,lstat返回的是该符号链接本身的信息;而stat返回的是该链接指向的文件的信息。(似乎有些晕吧,这样记,lstat比stat多了一个l,因此它是有本事处理符号链接文件的,因此当遇到符号链接文件时,lstat当然不会放过。而 stat系统调用没有这个本事,它只能对符号链接文件睁一只眼闭一只眼,直接去处理链接所指文件喽)
2 文件对应的属性
   struct stat {
        mode_t     st_mode;       //文件对应的模式,文件,目录等
        ino_t      st_ino;       //inode节点号
        dev_t      st_dev;        //设备号码
        dev_t      st_rdev;       //特殊设备号码
        nlink_t    st_nlink;      //文件的连接数
        uid_t      st_uid;        //文件所有者
        gid_t      st_gid;        //文件所有者对应的组
        off_t      st_size;       //普通文件,对应的文件字节数
        time_t     st_atime;      //文件最后被访问的时间
        time_t     st_mtime;      //文件内容最后被修改的时间
        time_t     st_ctime;      //文件状态改变时间
        blksize_t st_blksize;    //文件内容对应的块大小
        blkcnt_t   st_blocks;     //伟建内容对应的块数量
      };
可以通过上面提供的函数,返回一个结构体,保存着文件的信息.

3 获取文件属性的几个封装函数
///
// 功能说明 : 判断是否link文件
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_BOOL
// 使用说明 :
///
YCPP_BOOL IsLink ( const char * szDirName )
{
        struct stat st;

        return ( lstat (szDirName, &st) == 0 && S_ISLNK (st.st_mode) );
}

///
// 功能说明 : 判断是否block文件
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_BOOL
// 使用说明 :
///
YCPP_BOOL IsBlk ( const char * szDirName )
{
        struct stat st;

        return ( lstat (szDirName, &st) == 0 && S_ISBLK (st.st_mode) );
}

///
// 功能说明 : 判断是否FIFO文件
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_BOOL
// 使用说明 :
///
YCPP_BOOL IsFifo ( const char * szDirName )
{
        struct stat st;

        return ( lstat (szDirName, &st) == 0 && S_ISFIFO (st.st_mode) );
}

///
// 功能说明 : 判断是否目录文件
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_BOOL
// 使用说明 :
///
YCPP_BOOL IsDir ( const char * szDirName )
{
        struct stat st;

        return ( lstat (szDirName, &st) == 0 && S_ISDIR (st.st_mode) );
}

///
// 功能说明 : 判断是否设备文件
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_BOOL
// 使用说明 :
///
YCPP_BOOL IsDevice ( const char * szDirName )
{
        struct stat st;

        return ( lstat (szDirName, &st) == 0 && S_ISCHR (st.st_mode) );
}

///
// 功能说明 : 判断是否普通文件
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_BOOL
// 使用说明 :
///
YCPP_BOOL IsFile ( const char * szDirName )
{
        struct stat st;

        return ( lstat (szDirName, &st) == 0 && S_ISREG (st.st_mode) );
}

///
// 功能说明 : 判断是否可读
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_BOOL
// 使用说明 :
///
YCPP_BOOL canAccess ( const char *szFileName )
{
        if( access(szFileName, R_OK) == 0 )
                return YCPP_TRUE;

        return YCPP_FALSE;
}

///
// 功能说明 : 判断是否可修改
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_BOOL
// 使用说明 :
///
YCPP_BOOL canModify ( const char *szFileName )
{
        if ( access(szFileName, W_OK | R_OK) == 0 )
                return YCPP_TRUE;

        return YCPP_FALSE;
}

///
// 功能说明 : 获取文件上次修改时间
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_RET
// 使用说明 :
///
YCPP_RET FileLastMTime ( const char *szFileName )
{
        struct stat st;

        memset ( &st, 0, sizeof(struct stat) );
        if ( lstat (szFileName, &st) == 0 )
                return st.st_mtime;
       
        printf ( "%s FileLastMTime error %s\n", szFileName, strerror(errno) );
        return -1;
}

///
// 功能说明 : 获取文件大小,以字节为单位
// 输入参数 :
// 输出参数 : VOID
// 返回值   : YCPP_RET
// 使用说明 :
///
YCPP_RET FileSize ( const char *szFileName )
{
        struct stat st;
       
        memset ( &st, 0, sizeof(struct stat) );
        if ( lstat (szFileName, &st) == 0 )
                return st.st_size;
       
        printf ( "%s FileSize error %s\n", szFileName, strerror(errno) );
        return -1;
}




Logo

更多推荐