Linux中文件函数(二)
一、link、linkat、unlink、unlinkat、remove函数
创建一个指向现有文件的链接的方法是使用link函数或linkat函数。函数的原型为:
#include <unistd.h> int link(const char *existingpath,const char *newpath); int linkat(int efd,const char *existingpath,int nfd,const char *newpath,int flag);
这两个函数创建一个新目录项newpath,它引用现有文件existingpath。如果newpath已经存在,则返回错误。
只创建newpath中的最后一个分量,路径中的其它部分应当已经存在。
对于linkat函数,现有文件是通过efd和existingpath参数指定的。默认情况下,如果两个路径名中的任意一
个是相对路径,那么它需要通过相对于对应的文件描述符进行计算。如果两个文件描述符中的任一个设置为AT_F
DCWD,那么相应的路径名(如果它是相对路径)就通过相对于当前目录进行计算。如果任一路径名是绝对路径,
相应的文件描述符参数就会被忽略。当现有文件是符号链接时,由flag参数来控制linkat函数是创建指向现有符
号链接的链接还是创建指向现有符号链接所指向的文件的链接。如果在flag参数中设置了AT_SYMLINK_FOLLOW标志
,就是创建指向符号链接目标的链接。如果这个标志被清除了,则创建一个指向符号链接本身的链接。
为了删除一个现有的目录项,可以调用unlink函数。函数的原型为:
#include <unistd.h> int unlink(const char *pathname); int unlinkat(int fd,const char *pathname,int flag);
这两个函数删除目录项,并将由pathname所引用文件的链接计数减1。如果对该文件还有其它链接,则仍可以
通过其它链接访问该文件的数据。如果出错,则不对该文件做任何更改。
如果pathname参数是相对路径名,那么unlinkat函数计算相对于由fd文件描述符参数代表的目录的路径名。如
果fd参数设置为AT_FDCWD,那么通过相对于调用进程的当前工作目录来计算路径名。如果pathname参数是绝对路
径名,那么fd参数被忽略。flag参数给出了一种方法使调用进程可以改变unlinkat函数的默认行为。当AT_REMOVE
DIR标志被设置时,unlinkat函数可以类似于rmdir一样删除目录。如果这个标志被清除,unlinkat和unlink执行
同样的操作。
也可以用remove函数解除对一个文件或目录的链接。对于文件,remove的功能与unlink相同。对于目录,remo
ve的功能与rmdir相同。函数的原型为:
#include <stdio.h> int remove(const char *pathname);
二、rename、renameat函数
文件或目录可以用rename函数或者renameat函数进行重命名。函数的原型为:
#include <stdio.h> int rename(const char *oldname,const char *newname); int renameat(int oldfd,const char *oldname,int newfd,const char *newname);
(1)如果oldname指的是一个文件而不是目录,那么为该文件或符号链接重命名。在这种情况下,如果newname已
存在,则它不能引用一个目录。如果newname已存在,而且不是一个目录,则现将该目录项删除然后将oldname重命名
为newname。对包含oldname的目录以及包含newname的目录,调用进程必须具有写权限,因为将更改这两个目录。
(2)如果oldname指的是一个目录,那么为该目录重命名。如果newname已存在,则它必须引用一个目录,而且该
目录应当是空目录(空目录指的是该目录中只有.和..项)。如果newname存在(而且是一个空目录),则先将其删除
,然后将oldname重命名为newname。另外,当为一个目录重命名时,newname不能包含oldname作为其路径前缀。
(3)如果oldname或newname引用符号链接,则处理的是符号链接本身,而不是它所引用的文件。
(4)不能对.和..重命名。更确切的说,.和..都不能出现在oldname和newname的最后部分。
(5)如果oldname和newname引用同一文件,则函数不做任何更改而成功返回。如果newname已经存在,则调用进程
对它需要有写权限。另外,调用进程将删除oldname目录项,并可能要创建newname目录项,所以它需要对包含oldnam
e及包含newname的目录具有写和执行权限。
除了当oldname或newname指向相对路径名时,其它情况下renameat函数与rename函数功能相同。如果oldname参数
指定了相对路径,就相对于oldfd参数引用的目录来计算oldname。如果newname指定了相对路径,就相对于newfd引用
的目录来计算newname。oldfd或newfd参数都能设置为AT_FDCWD,此时相对于当前目录来计算相应的路径名。
三、symlink、symlinkat函数
可以用symlink或symlinkat函数创建一个符号链接。函数的原型为:
#include <unistd.h> int symlink(const char *actualpath,const char *sympath); int symlinkat(const char *actualpath,int fd,const char *sympath);
函数创建了一个指向actualpath的新目录项sympath。在创建此符号链接时,并不要求actualpath已经存在。并且
,actualpath和sympath并不需要位于同一文件系统中。
symlinkat函数与symlink函数类似,但sympath参数根据相对于打开文件描述符引用的目录(由fd参数指定)进行
计算。如果sympath参数指定的是绝对路径或者fd参数设置了AT_FDCWD值,那么symlinkat就等同于symlink函数。
open函数跟随符号链接,所以需要有一种方法打开该链接本身,并读该链接中的名字。readlink和readlinkat函数
提供了这种功能。函数的原型为:
#include <unistd.h> ssize_t readlink(const char *restrict pathname,char *restrict buf,size_t bufsize); ssize_t readlinkat(int fd,const char *restrict pathname,char *restrict buf,size_t bufsize);
两个函数组合了open、read和close的所有操作。如果函数执行成功,则返回读入buf的字节数。在buf中返回的符号
链接的内容不以null字节终止。当pathname参数指定的是绝对路径名或者fd参数的值为AT_FDCWD,readlinkat函数的行
为与readlink相同。但是,如果fd参数是一个打开目录的有效文件描述符并且pathname参数是相对路径名,则readlink
at计算相对于由fd代表的打开目录的路径名。
四、futimens、utimensat、utimes函数
一个文件的访问和修改时间可以用以下几个函数更改。futimens和utimensat函数可以指定纳秒级精度的时间戳。用
到的数据结构是与stat函数族相同的timespec结构。函数的原型为:
#include <sys/stat.h> int futimens(int fd,const struct timespec times[2]); int utimensat(int fd,const char *path,const struct timespec times[2],int flag);
这两个函数的times数组参数的第一个元素包含访问时间,第二个元素包含修改时间。时间戳可以按以下4种方式之
一进行指定:
(1)如果times参数是一个空指针,则访问时间和修改时间两者都设置为当前时间。
(2)如果times参数指定两个timespec结构的数组,任一数组元素的tv_nsec字段的值为UTIME_NOW,相应的时间戳就
设置为当前时间,忽略相应的tv_sec字段。
(3)如果times参数指向两个timespec结构的数组,任一数组元素的tv_nsec字段的值为UTIME_OMIT,相应的时间戳
保持不变,忽略相应的tv_sec字段。
(4)如果times参数指向两个timespec结构的数组,且tv_nsec字段的值为既不是UTIME_NOW也不是UTIME_OMIT,在这
种情况下,相应的时间戳设置为相应的tv_sec和tv_nsec字段的值。
执行这些函数所要求的优先权取决于times参数的值:
- 如果times是一个空指针,或者任一tv_nsec字段设置为UTIME_NOW,则进程的有效用户ID必须等于该文件的所有者
ID;进程对该文件必须具有写权限,或者进程是一个超级用户进程。
- 如果times是非空指针,并且任一tv_nsec字段的值既不是UTIME_NOW也不是UTIME_OMIT,则进程的有效用户ID必须
等于该文件的所有者ID,或者进程必须是一个超级用户进程。对文件只具有写权限是不够的。
- 如果times是非空指针,并且两个tv_nsec字段的值都为UTIME_OMIT,就不执行任何的权限检查。
futimens函数需要打开文件来更改它的时间,utimensat函数提供了一种使用文件名更改文件时间的方法。pathname
参数是相对于fd参数进行计算的,fd要么是打开目录的文件描述符,要么设置为特数值AT_FDCWD(强制通过相对于调用
进程的当前目录计算pathname)。如果pathname指定了绝对路径,那么fd参数被忽略。
utimensat的flag参数可用于进一步修改默认行为。如果设置了AT_SYMLINK_NOFOLLOW标志,则符号链接本身的时间
就会被修改(如果路径名指向符号链接)。默认的行为是跟随符号链接,并把文件的时间改成符号链接的时间。
#include <sys/time.h> int utimes(const char *pathname,const struct timeval times[2]);
utimes函数对路径名进行操作。times参数是指向包含两个时间戳(访问时间和修改时间)元素的数组的指针,两个
时间戳是用秒和微妙表示的。
五、mkdir、mkdirat、rmdir函数
用mkdir和mkdirat函数创建目录,用rmdir函数删除目录。函数原型为:
#include <sys/stat.h> int mkdir(const char *pathname,mode_t mode); int mkdirat(int fd,const char *pathname,mode_t mode);
这两个函数创建一个新的空目录。其中,.和..目录项是自动创建的。所指定的文件访问权限mode由进程的文件模式
创建屏蔽字修改。
mkdirat函数与mkdir函数类似。当fd参数具有特殊值AT_FDCWD或者pathname参数指定了绝对路径名时,mkdirat与mk
dir完全一样。否则,fd参数是一个打开目录,相对路径名根据此打开目录进行计算。
用rmdir函数可以删除一个空目录。空目录只包含.和..这两项的目录。函数的原型为:
#include <unistd.h> int rmdir(const char *pathname);
如果调用此函数使目录的链接计数成为0,并且也没有其它进程打开此目录,则释放此前目录占用的空间。如果在链
接计数达到0时,有一个或多个进程打开此目录,则在此函数返回前删除最后一个链接及.和..项。在此目录中不能在创
建新文件。但是在最后一个进程关闭它之前并不释放此目录。
六、chdir、fchdir、getcwd函数
每个进程都有一个当前工作目录,此目录是搜索所有相对路径名的起点。进程调用chdir或fchdir函数可以更改当前
工作目录。函数的原型为:
#include <unistd.h> int chdir(const char *pathname); int fchdir(int fd);
在这两个函数中,分别用pathname或打开文件描述符来指定新的当前工作目录。
从当前工作目录(.)开始,用..找到其上一级目录,然后读其目录项,直到该目录项中的i节点编号与工作目录i节点
编号相同,这样就找到了其对应的文件名。按照这种方法,逐层上移,直到遇到根,这样就得到了当前工作目录完整的
绝对路径名。函数getcwd提供了这种功能,函数原型为:
#include <unistd.h> char* getcwd(char *buf,size_t size);
必须向此函数传递两个参数,一个是缓冲区地址buf,另一个是缓冲区的长度size(以字节为单位)。该缓冲区必须
有足够的长度以容纳绝对路径名再加上一个终止null字节,否则返回出错。