今天给大家分享一篇关于100G文件0.2秒复制完的背后故事!
文章来源:本文转自公众号奇伢云存储
cp 引发的思考
cp
命令,把他给惊到了!背景是这样的:他用 cp
拷贝了一个 100 G的文件,竟然一秒不到就拷贝完成了!ls
看一把文件,显示文件确实是 100 G。sh-4.4# ls -lh
-rw-r--r-- 1 root root 100G Mar 6 12:22 test.txt
但是copy起来为什么会这么快呢?
sh-4.4# time cp ./test.txt ./test.txt.cp
real 0m0.107s
user 0m0.008s
sys 0m0.085s
cp
一秒没到就完成了工作,惊呆了,为啥呢?分析文件
我让他先用 du
命令看一下,却只有 2M ,根本不是100G,这是怎么回事?
sh-4.4# du -sh ./test.txt
2.0M ./test.txt
再看 stat
命令显示的信息:
sh-4.4# stat ./test.txt
File: ./test.txt
Size: 107374182400 Blocks: 4096 IO Block: 4096 regular file
Device: 78h/120d Inode: 3148347 Links: 1
Access: (0644/-rw-r--r--) Uid: ( 0/ root) Gid: ( 0/ root)
Access: 2021-03-13 12:22:00.888871000 +0000
Modify: 2021-03-13 12:22:46.562243000 +0000
Change: 2021-03-13 12:22:46.562243000 +0000
Birth: -
stat
命令输出解释:
-
Size 为 107374182400(知识点:单位是字节),也就是 100G ; -
Blocks 这个指标显示为 4096(知识点:一个 Block 的单位固定是 512 字节,也就是一个扇区的大小),这里表示为 2M;
划重点:
-
Size 表示的是文件大小,这个也是大多数人看到的大小; -
Blocks 表示的是物理实际占用空间;
同事问道:“文件大小和实际物理占用,这两个竟然不是相同的概念 !为什么是这样?”
“看来,我们必须得深入文件系统才能理解了,来,我给你好好讲讲。”
文件系统
现实的存取场景


划重点:存的时候必须记录一些关键信息(记录ID、给身份牌),取的时候才能正确定位到。
文件系统
-
登记名字就是在文件系统记录文件名;
-
生成的牌子就是元数据索引;
-
你的行李就是文件;
-
寄存室就是磁盘(容纳东西的物理空间);
-
管理员整套运行机制就是文件系统;
空间管理


-
先写数据:数据先按照 Block 粒度存储到磁盘的各个位置; -
再写元数据:然后把 Block 所在的各个位置保存起来,即inode(我用一本书来表示);

-
先读inode,找到各个 Block 的位置; -
然后读数据,构造一个完整的文件,给到用户;

inode/block 概念

-
前 12 个槽位(也就是 0 – 11 )我们成为直接索引; -
第 13 个位置,我们称为 1 级索引; -
第 14 个位置,我们称为 2 级索引; -
第 15 个位置,我们称为 3 级索引;

48K
,也就是说,48K 以内的文件,前 12 个槽位存储编号就能完全 hold 住。微信搜索公众号:网络安全与黑客技术,回复:黑客 领取资料 。4M(1024 * 4K)
空间 。
二级索引:
4G (4M/4 * 4K)
的空间。

所以,在这种文件系统(如ext2)上,通过这种间接块索引的方式,最大能支撑的文件大小 = 48K + 4M + 4G + 4T ,约等于 4 T。
这种多级索引寻址性能表现怎么样?
在不超过 12 个数据块的小文件的寻址是最快的,访问文件中的任意数据理论只需要两次读盘,一次读 inode,一次读数据块。访问大文件中的数据则需要最多五次读盘操作:inode、一级间接寻址块、二级间接寻址块、三级间接寻址块、数据块。
为什么cp那么快?
-
创建一个文件,这个时候分配一个 inode;
-
在 [ 0,4K ] 的位置写入 4K 数据,这个时候只需要 一个 block,把这个编号写到
block[0]
这个位置保存起来; -
在 [ 1T,1T+4K ] 的位置写入 4K 数据,这个时候需要分配一个 block,因为这个位置已经落到三级索引才能表现的空间了,所以需要还需要分配出 3 个索引块;
-
写入完成,close 文件;
实际存储如图:

重点:文件 size 只是 inode 里面的一个属性,实际物理空间占用则是要看用户数据放了多少个 block ,没写数据的地方不用分配物理block块。
总结
-
首先,最关键的是把磁盘空间切成离散的、定长的 block 来管理; -
然后,通过 inode 能查找到所有离散的数据(保存了所有的索引); -
最后,实现索引块和数据块空间的后分配;
本篇文章来源于微信公众号:程序IT圈
原创文章,作者:software,如若转载,请注明出处:https://www.sldh123.com/6443.html