什么是inode
inode译为中文就是索引节点,每个存储设备或存储设备的分区(存储设备是硬盘、软盘、U盘 … … )被格式化为文件系统后,应该有两部份,一部分是inode,另一部分是Block,Block是用来存储数据用的。而inode呢,就是用来存储这些数据的信息,这些信息包括文件大小、属主、归属的用户组、读写权限等。inode为每个文件进行信息索引,所以就有了inode的数值。操作系统根据指令, 能通过inode值最快的找到相对应的文件。
比如一本书,Block是它的每一页,而inode相当于他的目录,我们可以同各国目录快速的查找到某一页的内容。
inode的内容
我们可以通过stat
指令查看某一文件的inode信息
inode中的信息
- 文件的字节数,块数
- 文件拥有者的User ID
- 文件的Group ID
- 文件的读、写、执行权限
- 文件的时间戳,共有三个:ctime指inode上一次变动的时间,mtime指文件内容上一次 变动的时间,atime指文件上一次打开的时间。
- 链接数,即有多少文件名指向这个inode
- 文件数据block的位置
- inode编号(对应数组的下标)
inode的大小
每个inode节点的大小,一般是128字节或256字节。inode节点的总数,在格式化时就给定,一般是每1KB或每2KB就设置一个inode。
可以用df
命令查看每个硬盘分区的inode总数和已经使用的数量
查看每个inode节点的大小,可以用如下命令:
sudo dumpe2fs -h /dev/sda5 | grep "Inode size"
由于每个文件都必须有一个inode,因此有可能发生inode已经用光,但是硬盘还未存满的情况。这时,就无法在硬盘上创建新文件。
inode号码
每个inode都有一个号码,操作系统用inode号码来识别不同的文件。Unix/Linux系统内部不使用文件名,而使用inode号码来识别文件。对于系统来说,文件名只是inode号码便于识别的别称或者绰号。表面上,用户通过文件名,打开文件。
可以用ls -i
命令来查看文件的inode号码
目录文件
打开目录,实际上就是打开目录文件。
每个目录项,由两部分组成:所包含文件的文件名,以及该文件名对应的inode号码。
ls
命令只列出目录文件中的所有文件名:
ls -i
命令可以查看文件名和文件的inode号
如果要查看文件的详细信息,就必须根据inode号码,访问inode节点,读取信息。ls -l
命令列出文件的详细信息。
目录文件的读权限(r)和写权限(w),都是针对目录文件本身。由于目录文件内只有文件名和inode号码,所以如果只有读权限,只能获取文件名,无法获取其他信息,因为其他信息都储存在inode节点中,而读取inode节点内的信息需要目录文件的执行权限(x)。
硬链接和软链接
什么是硬链接
在Linux 文件系统中,inode值相同的文件是硬链接文件,也就是说,不同的文件名,inode可能是相同的,一个inode值可以对应多个文件。理解链接文件并不难,看看例子就会了。在Linux中,链接文件是通过ln
工具来创建的。
(1) 硬链接,以文件副本的形式存在。但不占用实际空间。
(2) 不允许给目录创建硬链接。
(3) 硬链接只有在同一个文件系统中才能创建。
(4) 删除其中一个硬链接文件并不影响其他有相同 inode 号的文件。
什么是软链接
类同与windos的快捷方式,给文件创建一个快速的访问路径,它依赖于原文件,与普通文件没什么不同,inode 都指向同一个文件在硬盘中的区块。当原文件出现问题后,该链接不可用。ln -s
命令可以创建软链接。
(1)可以应用于目录
(2)可以跨文件系统
(3)不会增加被链接文件的链接次数
(4)大小为指定的绝对路径所包含的字符总数
(5)有自己的inode号
(6)权限无关紧要
借鉴了一个代码
#include <stdio.h>
#include <unistd.h>
int main(int argc, char *argv[])
{
int ret;
if(argc < 5)
{
printf("please check the input file!");
return 1;
}
//硬链接
ret = link(argv[1], argv[2]);
if(ret)
{
printf("link failed!");
return 0;
}
printf("link %s to %s success", argv[1], argv[2]);
//软连接
ret = symlink(argv[3], argv[4]);
if(ret)
{
printf("symlink failed!");
return 0;
}
printf("symlink %s to %s success", argv[3], argv[4]);
return 0;
}
在linux中编译程序:gcc link.c -o link
后,生成名为link的文件,再运行
./link good good1 bad bad1 命令,用两种连接生成新的文件,具体如下图:
用硬连接生成的是good1,用软连接生成的是bad1,很明显,硬链接生成的文件的文件标识与源文件一样,而软连接的标识不一样。硬连接生成的文件与源文件一摸一样,对任何一个文件改动,另一个也跟着变动;软连接生成的文件相当于源文件的快捷方式。
#include <stdio.h>
#include <unistd.h>
int main(int argc, char *argv[])
{
int ret;
if(argc < 3)
{
printf("please check the file!\n");
return 1;
}
//删除硬链接
ret = unlink(argv[1]);
if(ret)
{
printf("unlink %s failed!\n", argv[1]);
return 1;
}
printf("unlink %s success!\n", argv[1]);
//删除软链接
ret = unlink(argv[2]);
if(ret)
{
printf("unlink %s failed!\n", argv[2]);
return 1;
}
printf("unlink %s success!\n", argv[2]);
return 0;
}
在上面的基础上调用删除连接的函数删除连接的程序
通过inode初步认识文件系统
inode相当于文件的一个目录,我们额可以通过查看文件的inode知道文件的大小,位置等信息,方便我们查找,有时我们也可以通过删除inode号来删除文件。系统内部这个过程分成三步:首先,系统找到这个文件名对应的inode号码;其次,通过inode号码,获取inode信息;最后,根据inode信息,找到文件数据所在的block,读出数据。