目录和硬链接
在文件系统中,目录的存储方式类似于普通文件,区别有:
- 在目录的i-node条目中,会将其标记为一种不同的文件类型
- 目录是经特殊组织而成的文件。本质上就是一个表格,包含文件名和i-node编号。
进程能够打开一个目录,但却不能用read() 去读取目录的内容。同样也不能使用write() 来改变目录内容。仅能借助于open() 、link() 、mkdir() 、symlink() 、unlink() 以及rmdir() 之类的系统调用来间接改变其内容。
i-node表的编号始于1,而非0,因为若目录条目的i-node字段值为0,则表明该条目尚未使用。
能够在相同或者不同目录中创建多个名称,每个均指向相同的i-node节点。也将这些名称称为链接。有时也称为硬链接。 可在shell中利用ln 命令为一个已经存在的文件创建新的硬链接:
$ln <file> <link>
- 仅当i-node的链接技术降为0时,也就是移除了文件的所有名字时,才会删除文件的i-node记录和数据块。
- 同一文件的所有名字(链接)地位平等,没有一个名字会优于其他名字。在移除与文件相关的第一个名称后,物理文件继续存在,直到移除所有名字。
对硬链接的限制:
- 因为目录条目(硬链接)对文件的指代采用了i-node编号,而i-node编号的唯一性仅在一个文件系统之内才能得到保障,所以硬链接必须与其所指代的文件驻留在同一文件系统中。
- 不能为目录创建硬链接,从而避免出现诸多令诸多系统程序陷于混乱的链接环路。
符号链接(软链接)
符号链接,也称为软链接,是一种特殊的文件类型,其数据是另一文件的名称。 在shell中,符号链接是由ln -s 命令创建的。ln -F 命令输出结果中会在符号链接的尾部标记@ 。
符号链接的内容既可以是绝对路径,也可以是相对路径。解释相对符号链接时可以链接本身的位置作为参照点。
符号链接的地位不如硬链接。尤其是,文件的链接计数中未将符号链接计算在内。 因此,如果移除了符号链接所指向的文件名,符号链接本身还将继续存在,尽管无法解引用,也将此类链接称之为悬空链接。甚至,可以为并不存在的文件名创建一个符号链接。
因为符号链接指代一个文件名,而非i-node编号,所以可以用其来链接不同文件系统中的一个文件。也可以为目录茶ungj符号链接。
符号链接之间可能会形成链路。当在各个文件相关的系统调用中指定了符号链接时,内核会对一系列链接层层解去引用,直抵最终文件。(Linux将对一个完整路径名的解引用此书限制为40次。防止内核代码在解析符号链接时引发堆栈溢出。)
系统调用对符号链接的解释
许多系统调用都会对符号链接进行解引用处理,从而对链接所指向的文件展开操作。还有一些系统调用对符号链接则不作处理,直接操作于链接文件本身。总结如下:
函数 | 是否对链接解引用 | 备注 |
---|
access() | | | acct() | | | bind() | | UNIX域套接字带有路径名 | chdir() | | | chmod() | | | chown() | | | chroot() | | | creat() | | | exec() | | | getxattr() | | | lchown() | N | | lgetxattr | N | | link() | N | | listxattr() | | | llistxattr() | N | | lremovexattr() | N | | lsetxattr | N | | lstat() | N | | lutimes() | N | | open() | | 除非指定了O_NOFOLLOW 或者O_EXCL|O_CREAT | opendir() | | | pathconf() | | | pivot_root() | | | quotactl() | | | readlink() | N | | removexattr() | | | rename() | N | 无论那个参数中的链接,都不会对其进行解引用 | rmdir() | N | 若参数为符号链接,则调用失败,并将errno 置为ENOTDIR | setxattr() | | | stat() | | | statfs(), statvfs() | | | swapon(), swapoff() | | | truncate() | | | unlink() | N | | uselib() | | | utime(), utimes() | | |
参考资料:《Linux/UNIX系统编程手册》
|