首页 > 代码库 > 文件和目录1(文件属性和权限)

文件和目录1(文件属性和权限)

获取文件属性(struct stat):<sys/stat.h>
int stat(const char* restrict pathname,struct stat* restrict buf);int fstat(int fd,struct stat* restrict buf);int lstat(const char* restrict pathname,struct stat* restrict buf); //获取软连接的文件属性
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; //文件所有者uid    gid_t st_gid; //文件所有者gid    off_t st_size; //文件大小    time_t st_atime; //access time    time_t st_mtime; //文件数据modification time    注意:acsess和stat函数不更改这三个时间    time_t st_ctime; //属性最近一次change time    blksize_t st_blksize; //block size    blkcnt_t st_blocks; //blocks};
<1>文件类型宏函数: 参数都是st_mode            <sys/stat.h>
S_ISREG() //判断是否是普通文件  S_ISDIR() S_ISCHR()  S_ISBLK()  S_ISFIFO()  S_ISLNK()  S_ISSOCK()   //套接字
<2>文件访问权限位和设置用户ID,设置组ID位都包含在st_mode中,st_mode屏蔽字如下(都是常量):
S_ISUID(设置uid位)  S_ISGID(设置gid位)  S_ISVTX(粘贴位)S_IRWXU  S_IRWXG   S_IRWXO  //可读写执行S_IRUSR   S_IWUSR    S_IXUSRS_IRGRP   S_IWGRP    S_IXGRPS_IROTH   S_IWOTH    S_IXOTH

 

文件访问权限说明

<1>进程相关的用户ID和组ID

RUID RGID:在登录时取自口令文件中的登录项,只有超级用户进程可以改变EUID EGID:决定了访问权限。SUID SGID(保存的设置用户ID和保存的设置组ID):在执行一个程序时保存EUID EGID的副本。通过测试常量POSIX_SAVED_IDS  或 调用sysconf( _SC_SAVED_IDS做参数)判断系统是否支持这种特征。
a,当执行一个程序文件时,EUID等于RUID,EGID等于RGID;
b,如果设置了设置用户ID位和设置组ID位时,当执行此文件时,EUID为文件的用户ID(st_uid),EGID为文件的组ID(st_gid)
<2>对文件的各种操作所需的权限
a,打开任意类型的文件(有可执行权限):进程的EUID等于文件的st_uid或者是gid匹配或者进程的EUID为0(超级用户);
b,删除文件(对包含该文件的目录有写和执行权限):进程EUID等于文件的st_uid后者是gid匹配或者进程的EUID为0(超级用户);
c,在目录下创建文件(对该目录有写和执行权限):
新文件的用户ID设置为进程的EUID;新文件的组ID可以是EGID 或者是它所在目录的组ID(当它所在目录的设置gid位被设置)

 

测试访问权限

// <unistd.h>int access(const char *pathname, int mode)    //access是进程按照其RUID DGID来测试访问权限的,而open是按照EUID EGID//mode参数:R_OK     W_OK      X_OK     F_OK(文件是否存在)mode_t umask(mode_t cmask)//为进程设置文件模式创建屏蔽字,并返回以前的值。进程在创建新文件或目录时,cmask中为1的位在新文件mode中相应的位则被关闭//cmask参数:是个文件访问权限位按位或构成的或者是 0400(用户读)的形式

 

更改现有文件访问权限

// <sys/stat.h>int chmod (const char *pathname, mode_t mode);int fchmod(int filedes, mode_t mode);/*a,参数mode:是st_mode,先要用stat,fstat,lstat获取文件的struct stat    mode一般是这种格式:statbud.st_mode & ~S_IXGRP | S_ISGID  设置文件的设置组ID位,取消文件的所有组可执行权限*//*b,chmod在一定条件下自动清除两个权限位:如果是非超级用户试图设置粘住位,那么粘住位会被清除,只有root能设置普通文件的粘住位如果新创文件gid不等于进程egid或进程附加组id,并且是非超级用户,那么设置组ID位会被清除*/

 

更改文件的uid和gid

//<unistd.h>int chown (const char *pathname, uid_t owner, gid_t group);int lchown (const char *pathname, uid_t owner, gid_t group);int fchown (int filedes, uid_t owner, gid_t group);//参数:owner或group为-1或为文件的uid gid时,对应的uid gid不变/*若_POSIX_CHOWN_RESTRICTED对指定的文件起作用(定义在<unistd.h>可用pathconf/fpathconf函数查询):<1>只有root进程才允许更改uid.<2>非root进程改变gid的条件:euid等于文件uid(进程拥有此文件);或者文件uid不变,参数group等于进程euid或附加组ID之一*///同时需要注意的是,如果这些函数由非超级用户调用,则在成功返回时,设置uid和gid位都会被清除

 

文件截断

文件长度:st_size(文件长度) 接近 st_blksize(文件使用块大小)和st_blocks(块数)的乘积。
当文件有空洞时st_size可能很大,而实际占用磁盘空间较少
文件截短:int truncate(const char* filename,off_t length);   int ftruncate(int fd,off_t length);   <unistd.h>
如果length比原来文件短的话,那么文件在length偏移之后数据就不可以访问了。如果length比 原来文件长的话,那么会创造一个空洞出来
 

符号链接

 

//<unistd.h>int link(const char *exitingpath, const char *newpath)  //原子操作:创建一个新目录项(即链接) ,链接计数加1int unlink(const chat *pathname)   //删除一个目录项,链接计数减1

 

a,解除对文件的链接的条件:对包含该目录项的目录有写和执行权限;若设置的粘住位,需满足以下条件之一:拥有该文件,拥有该目录,是root用户  
b,文件链接数目为0时,文件就会被删除,因此,unlink可以删除文件;
        但当st_nlink=0时仍有进程访问该文件,那么磁盘空间不会被释放,该文件仍然可写。
不跟随符号链接的函数(对链接本身起作用):
函数不跟随链接跟随链接
access
 
Y
chdir
 
Y
chmod
 
Y
chown
 
Y
creat
 
Y
exec
 
Y
lchownY
 
link
 
Y
lstatY
 
open
 
Y
opendir
 
Y
pathconf
 
Y
readlinkY
 
removeY
 
renameY
 
stat
 
Y
truncate
 
Y
unlinkY
 
 
symlink创建符号链接
readlink打开符号链接并读该链接中的名字,open跟随符号链接
 
更改文件的st_atime和st_mtime     <utime.h>
int utime(const char *pathname, const struct utimebuf *times);
参数times为空时,则两个时间设为当前时间
struct utimebuf{
time_t atime;
time_t modtime;
}
 
目录操作
int mkdir(const char *pathname, mode_t mode);          <sys/stat.h>
读目录  <dirent.h>
DIR *opendir(const char *pathname);
struct dirent *readdir(DIR *dp);
rewinddir  closedir  telldir  seekdir 
更改当前的工作目录    <unistd.h>
int chdir(const char *pathname);
int fchdir(int fd);
char *getcwd(char *buf, size_t size); // 获取当前动作目录 参数是缓冲区地址和长度
 
特殊设备文件
st_dev设备号   参数是st_dev : major()  minor()
sr_rdev只有字符特殊文件和块特殊文件才有这个值,表示实际设备的设备编号。

 

文件和目录1(文件属性和权限)