爱程序网

文件和目录之link、unlink、remove和rename函数

来源: 阅读:

任何一个文件可以有多个目录项指向其i节点。创建一个指向现有文件的链接的方法是使用link函数。

#include <unistd.h>
int link( const char *existingpath, const char *newpath );
返回值:若成功返回0,若出错返回-1

此函数创建一个新目录项newpath,它引用现有的文件existingpath。如若newpath已经存在,则返回出错。只创建newpath中的最后一个分量,路径中的其他部分应当已经存在。

创建新目录项以及增加链接计数应当是个原子操作。虽然POSIX.1允许实现支持跨文件系统的链接,但是大多数实现要求这两个路径名在同一个文件系统中。如果实现支持创建指向一个目录的硬链接,那么也仅限于超级用户才可以这样做。其理由是这样做可能在文件系统中形成循环,大多数处理文件系统的实用程序都不能处理这种情况。因此很多文件系统实现不允许对目录的硬链接。

为了删除一个现有的目录项,可以调用unlink函数。

#include <unistd.h>
int unlink( const char *pathname );
返回值:若成功则返回0,若出错则返回-1

此函数删除目录项,并将由pathname所引用文件的链接计数减1。如果还有指向该文件的其他链接,则仍可通过其他链接访问该文件的数据。如果出错,则不对该文件做任何更改。

为了解除对文件的链接,必须对包含该目录项的目录具有写和执行的权限。如果对该目录设置了粘住位,则对该目录必须具有写权限,并且具备下面三个条件之一:

拥有该文件。

拥有该目录。

具有超级用户特权。

只有当链接计数达到0时,该文件的内容才可被删除。另一个条件也会阻止删除文件的内容——只要有进程打开了该文件,其内容也不能删除。关闭一个文件时,内核首先检查打开该文件的进程数。如果该数达到0,然后内核检查其链接数,如果这个数也是0,那么就删除该文件的内容。

程序清单4-5 打开一个文件,然后unlink它

[root@localhost apue]# cat prog4-5.c
#include "apue.h"
#include <fcntl.h>

int 
main(void)
{
        if(open("tempfile", O_RDWR) < 0)
                err_sys("open error");
        if(unlink("tempfile") < 0)
                err_sys("unlink error");
        printf("file unlinked\n");
        sleep(15);
        printf("done\n");
        exit(0);
}

运行该程序,其结果是:

[root@localhost apue]# ls -l tempfile
-rw-r--r-- 1 root root 38117937 01-02 18:18 tempfile
[root@localhost apue]# df /home
文件系统               1K-块        已用     可用      已用% 挂载点
/dev/sda3         15289132   
209228
  14290712   2% /home
[root@localhost apue]# ./prog4-5 &
[1] 5152
[root@localhost apue]# file unlinked
[root@localhost apue]# ls -l tempfile
ls: tempfile: 没有那个文件或目录                                                                         此时,目录项已删除
[root@localhost apue]# df /home
文件系统               1K-块        已用     可用      已用% 挂载点
/dev/sda3         15289132    209228  14290712   2% /home                                   从已用/可用空间可以看出该文件内容还没有删除
[root@localhost apue]# done
[root@localhost apue]# df /home
文件系统               1K-块        已用     可用      已用% 挂载点
/dev/sda3         15289132    171956  14327984   2% /home                                   睡眠结束后,进程结束,此时才删除了该文件内容

unlink的这种性质经常被程序用来确保即使是在该程序崩溃时,它所创建的临时文件也不会遗留下来。进程用open或creat创建一个文件,然后立即调用unlink。因为该文件仍旧是打开的,所以不会将其内容删除。只有当进程关闭该文件或终止时(在这种情况下,内核会关闭该进程打开的全部文件),该文件的内容才会被删除。

如果pathname是符号链接,那么unlink删除该符号链接,而不会删除由该链接所引用的文件。给出符号链接名情况下,没有一个函数能删除由该链接所引用的文件。

超级用户可以调用unlink,其参数pathname指定一个目录,但是通常应当使用rmdir函数,而不是使用这种方式。

我们也可以用remove函数解除对一个文件或目录的链接。对于文件,remove的功能与unlink相同。对于目录,remove的功能与rmdir相同。

#include <stdio.h>
int remove( const char *pathname);
返回值:若成功则返回0,若出错则返回-1

文件或目录用rename函数更名。

#include <stdio.h>
int rename( const char *oldname, const char *newname );
返回值:若成功则返回0,若出错则返回-1

根据oldname是指文件还是目录,有几种情况要加以说明。我们也应说明如果newname已经存在将会发生什么。

(1)如果oldname指的是一个文件而不是目录,那么为该文件或符号链接更名。在这种情况下,如果newname已存在,则它不能引用一个目录。如果newname已存在,而且不是一个目录,则先将该目录项删除然后将oldname更名为newname。对包含oldname的目录以及包含newname的目录,调用进程必须具有写权限,因为将更改这两个目录。

(2)如果oldname指的是一个目录,那么为该目录更名。如果newname已存在,则它必须引用一个目录,而且该目录应当是空目录(空目录指的是该目录中只有.和..项)。如果newname存在(而且是一个空目录),则先将其删除,然后将oldname更名为newname。另外,当为一个目录更名时,newname不能包含oldname作为其路径前缀。例如,不能将/usr/foo更名为/usr/foo/testdir,因为旧名字(/usr/foo)是新名字的路径前缀,因而不能将其删除。

(3)如若oldname或newname引用符号链接,则处理的是符号链接本身,而不是它所引用的文件。

(4)作为一个特例,如果oldname和newname引用同一文件,则函数不做任何更改而成功返回。

如若newname已经存在,则调用进程对它需要有写权限(如同删除情况一样)。另外,调用进程将删除oldname目录项,并可能要创建newname目录项,所以它需要对包含oldname以及newname的目录具有写和执行权限。

本篇博文内容摘自《UNIX环境高级编程》(第二版),仅作个人学习记录所用。关于本书可参考:http://www.apuebook.com/

 

 

关于爱程序网 - 联系我们 - 广告服务 - 友情链接 - 网站地图 - 版权声明 - 人才招聘 - 帮助