UNIX环境编程学习笔记(10)——文件I/O之硬链接和符号链接

lienhua34
2014-09-15

1 文件系统数据结构

UNIX 文件系统通过 i 节点来存储文件的信息。如图 1 所示为一个磁盘柱面上的 i 节点和数据块示意图。其中 i 节点是一个固定长度的记录项,它包含了有关文件的大部分信息。数据块用于存储文件的实际内容。每个文件的 i 节点会记录该文件的内容所占用的数据块信息。

图 1: i 节点和数据块

图 1 中还有一些信息需要进行说明:

1. 每个目录项只存储了文件的文件名和 i 节点编号(每个文件系统各自对它们的 i 节点进行编号)。文件的其它信息则记录在 i 节点中,例如,文件类型、文件访问权限位、文件长度等。

2. 每个 i 节点都有一个链接计数,其值是指向该 i 节点的目录项数。只有当链接计数减少至 0 时,才可删除该文件(即释放该文件所占用的数据块)。通过 i 节点链接使多个目录项指向同一个文件的这种链接类型称为硬链接

2 硬链接和符号链接

上一节讲到了,硬链接是通过 i 节点编号来使多个目录项指向同一个文件。因此,硬链接存在一些限制,

• 硬链接要求链接和文件位于同一个文件系统中(因为每个文件系统具有各自的 i 节点编号)。

• 只有超级用户才能创建指向目录的硬链接(避免在文件系统中存在循环)。

对于符号链接,该文件的实际内容(在数据块中)包含了该符号链接所指向的文件的名字。对符号链接以及它所指向何种对象并无任何文件系统限制,任何用户都可以创建指向目录的符号链接。符号链接一般用于将一个文件或整个目录结构移到文件系统中的另一个位置。

3 创建链接 link 函数、解除链接 unlink 函数

link 函数用于创建一个现有文件的硬链接。

#include <unistd.h>

int link(cosnt char *existingpath, const char *newpath);

返回值:若成功则返回0,若出错则返回-1。

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

unlink 函数删除一个现有的目录项。

#include <unistd.h>

int unlink(const char *pathname);

返回值:若成功则返回0,若出错则返回-1.

此函数删除目录项,比将由 pathname 所引用的文件的链接计数减 1。如果出错,则不对该文件做任何修改。只有当链接计数达到 0 时,该文件的内容才会被删除。另一个阻止删除文件内容的条件是:有进程打开着该文件。关闭一个文件时,内核首先检查打开该文件的进程数。如果该数达到 0,然后内核检查其链接计数,如果链接计数也是 0,那么就删除该文件的内容。

如果 pathname 是符号链接,那么 unlink 删除该符号链接,而不会删除该符号链接所引用的文件。

例子:

下面程序先使用 open 打开文件 tempfile,然后 unlink 删除目录项,接着进程进入 15 秒钟睡眠时间。

#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <fcntl.h>
int
main(void)
{
    if (open("tempfile", O_RDWR) < 0) {
        printf("open error: %s\n", strerror(errno));
        exit(-1);
    }
    if (unlink("tempfile") < 0) {
        printf("unlink error: %s\n", strerror(errno));
        exit(-1);
    }
    printf("file unlinked\n");
    sleep(15);
    printf("done\n");
    exit(0);
}

编译该程序,生成文件 unlinkdemo,然后执行该文件,

 1 lienhua34:demo$ ls -l tempfile
 2 -rw-r--r-- 1 lienhua34 lienhua34 20094393  9月 15 21:06 tempfile
 3 lienhua34:demo$ df /home
 4 文件系统           1K-块     已用      可用 已用% 挂载点
 5 /dev/sda1      305603832 14811828 275268216    6% /
 6 lienhua34:demo$ ./unlinkdemo &
 7 [1] 3113
 8 lienhua34:demo$ file unlinked
 9 ls -l tempfile
10 ls: 无法访问tempfile: 没有那个文件或目录
11 lienhua34:demo$ df /home
12 文件系统           1K-块     已用      可用 已用% 挂载点
13 /dev/sda1      305603832 14811828 275268216    6% /
14 lienhua34:demo$ done
15 df /home
16 文件系统           1K-块     已用      可用 已用% 挂载点
17 /dev/sda1      305603832 14792140 275287904    6% /

通过上面的执行结果,我们可以看出,只有等到执行 unlinkdemo 文件的进程结束之后(第14行),文件 tempfile 的内容才被真正删除。unlink 的这种性质经常被程序用来确保即使是在该程序崩溃时,它所创建的临时文件也不会遗留下来。进程用 open 或 creat 创建一个文件,然后立即调用 unlink。因为该文件被进程打开着,所以不会将其内容删除。只有当进程关闭该文件或终止时(在这种情况下,内核关闭该进程打开的全部文件),该文件的内容才会被删除。

4 符号链接的 symlink 和 readlink 函数

symlink 函数创建一个符号链接。

#include <unistd.h>

int symlink(const char *actualpath, const char *sympath);

返回值:若成功则返回0,若出错则返回-1。

该函数创建了一个指向 actualpath 的新目录项 sympath,在创建此符号链接时,并不要求 actualpath 已经存在。并且,actualpath 和 sympath并不需要位于同一文件系统中。

readlink 函数打开符号链接本身,并读取该链接中的内容(不是该链接所引用的文件的内容)。

#include <unistd.h>

ssize_t readlink(const char *restrict pathname, char *restrict buf, size_t bufsize);

返回值:若成功则返回读到的字节数,若出错则返回-1。

如果此函数成功执行,则返回读入 buf 的字节数。在 buf 中返回的符号链接的内容不以 null 字符终止。

例子:

下面程序创建文件 bar 的符号链接 barlink,然后读入符号链接 barlink的内容并进行打印。

#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#define BUFFER_LEN 2014
int
main(void)
{
    char buf[BUFFER_LEN];
    ssize_t buflen;
    if (symlink("bar", "barlink") < 0) {
        printf("symlink error: %s\n", strerror(errno));
        exit(-1);
    }
    printf("create symbol link \"barlink\".\n");
    if ((buflen = readlink("barlink", buf, BUFFER_LEN)) < 0) {
        printf("readlink error: %s\n", strerror(errno));
        exit(-1);
    }
    printf("barlink context: %s\n", buf);
    exit(0);
}

编译该程序,生成 symlinkdemo,然后运行该文件,

lienhua34:demo$ gcc -o symlinkdemo symlinkdemo.c
lienhua34:demo$ ./symlinkdemo
create symbol link "barlink".
barlink context: bar
lienhua34:demo$ ls -l bar barlink
-rw-r--r-- 1 lienhua34 lienhua34 7
9月 15 20:02 bar
lrwxrwxrwx 1 lienhua34 lienhua34 3 9月 15 20:07 barlink -> bar
lienhua34:demo$ cat bar
in bar

(done)

时间: 2024-11-08 22:28:03

UNIX环境编程学习笔记(10)——文件I/O之硬链接和符号链接的相关文章

UNIX环境编程学习笔记(13)——文件I/O之标准I/O流

lienhua342014-09-29 1 标准 I/O 流 之前学习的都是不带缓冲的 I/O 操作函数,直接针对文件描述符的,每调用一次函数可能都会触发一次系统调用,单次调用可能比较快捷.但是,对于需要频繁进行 I/O 操作的程序,频繁触发系统调用产生的消耗太大. 标准 I/O 库提供了带缓冲的 I/O 操作函数,这些函数围绕着一种叫做流(stream)的东西进行.当使用标准 I/O 库打开或创建一个文件时,系统提供了一个流与这个文件相关联.通过流的读入和输出完成所需要的 I/O操作. 标准

UNIX环境编程学习笔记(4)——文件I/O之dup复制文件描述符

lienhua342014-08-23 UNIX 提供了两个函数 dup 和 dup2 用于复制一个现存的文件描述符. #include <unistd.h> int dup(int filedes); int dup2(int filedes, int filedes2); 返回值:若成功则返回新的文件描述符,如出错则返回-1. 由 dup 函数返回的文件描述符一定是当前可用文件描述符中的最小描述符.用 dup2 函数则可以通过参数 filedes2 指定目标文件描述符.如果filedes2

UNIX环境编程学习笔记(6)——文件I/O之判断文件类型

lienhua342014-09-01 1 文件类型 我们平时最常接触的文件类型有普通文件(regular file)和目录(di-rectory file),但是 UNIX 系统提供了多种文件类型: (1) 普通文件(regular file) 这种文件包含了某种形式的数据,这些数据无论是文件还是二进制对于 UNIX 内核而言都是一样的.对普通文件内容的解释有处理该文件的应用程序进行. (2) 目录文件(directory file) 目录文件包含了其他文件的名字以及指向与这些文件有关信息的指

UNIX环境编程学习笔记(9)——文件I/O之文件访问权限的屏蔽和更改

lienhua342014-09-10 1 文件访问权限 在文件访问权限和进程访问控制中,我们已经讲述过文件访问权限位,为了方便,我们重新列在下面, 表 1: 文件的 9 个访问权限位  st_mode 屏蔽  意义  S_IRUSR  用户 -读  S_IWUSR  用户 -写  S_IXUSR  用户 -执行  S_IRGRP   组 -读  S_IWGRP  组 -写  S_IXGRP  组 -执行  S_IROTH  其他 -读  S_IWOTH  其他 -写  S_IXOTH  其他

UNIX环境编程学习笔记(7)——文件I/O之文件访问权限与进程访问控制

lienhua342014-09-02 1 文件的设置用户 ID位 和设置组 ID位 与进程相关联的 ID 如下表所示, 表 1: 与进程相关联的用户 ID 和组 ID 实际用户 ID 我们实际上是谁 实际组 ID 有效用户 ID 用于文件访问权限检查 有效组 ID 附加组 ID 保存的设置用户 ID 由 exec 函数保存 保存的设置组 ID 保存的设置用户 ID 和保存的设置组 ID 在执行一个程序时包含了有效用户 ID 和有效组 ID 的副本,这个后面我们学习到进程时在详细学习. 此处,我

UNIX环境编程学习笔记(3)——文件I/O之内核 I/O 数据结构

lienhua342014-08-27 内核使用三种数据结构表示打开的文件,分别是文件描述符表.文件表和 V 节点表. (1) 每个进程在进程表中都有一个记录项,记录项中包含有一张打开文件描述符表,每个描述符占用一项.与每个文件描述符相关联的是: (a) 文件描述符标志. (b) 指向一个文件表项的指针. (2) 内核为所有打开文件维持一张文件表.每个文件表项包含: (a) 文件状态标志(读.写.添写.同步和非阻塞等). (b) 当前文件偏移量. (c) 指向该文件 V 节点表项的指针. (3)

UNIX环境编程学习笔记(12)——文件I/O之目录操作

lienhua342014-09-18 1 引言 在 UNIX 系统中,目录是一种特殊的文件类型.我们可以使用 open 函数来打开目录,获取文件描述符,然后调用 stat 函数来获取目录的属性信息,但是我们却不能够使用 read 函数来读取目录内容.例如,下面例子所示, #include <stdlib.h> #include <stdio.h> #include <errno.h> #include <string.h> #include <fcn

UNIX环境编程学习笔记(8)——文件I/O之校验当前登录用户对文件的访问权限

lienhua342014-09-03 通过前面一篇随笔(文件访问权限与进程访问控制),我们知道内核校验文件的访问权限使用的是进程的有效用户 ID 和有效组 ID.但有时我们需要知道当前登录用户对某个文件访问权限.虽然说进程的有效用户 ID 和有效组 ID 通常分别等于当前登录用户 ID 和用户所在组 ID.例如,一个进程可能因设置用户 ID 以另一个用户权限运行,它仍可能想验证当前实际登录的用户是否能否访问一个给定的文件. access 函数提供了按照实际用户 ID 和实际组 ID 进行访问权

UNIX环境编程学习笔记(11)——文件I/O之文件时间以及 utime 函数

lienhua342014-09-16 1 文件的时间 每个文件都有三个时间字段,如表 1 所示. 表 1: 文件的三个时间字段 说明 字段 st_atime 文件数据的最后访问时间 st_mtime 文件数据的最后修改时间 st_ctime i 节点状态的最后更改时间 最后修改时间是文件内容最后一次被修改的时间.更改状态时间是该文件的 i 节点最后一次被修改的时间. 2 utime 函数 utime 函数提供了对文件最后访问时间和最后修改时间的更改功能. #include <utime.h>