前提知识
当我们的文件没有被打开的时候,只能在磁盘放着,磁盘上有大量的文件,这么多文件,也是要被静态管理起来的,方便我们随时打开。
1.1.磁盘的物理结构
1.磁盘是我们计算机中唯一的一个机械结构。
2.磁盘是一个机械结构,而且还是一个外设,所以相比一CPU他就很慢很慢。
3.现在我们使用的笔记本电脑,已经很少用磁盘(HDD)了。用的是固态硬盘(SSD)。
4.在企业端,磁盘依旧是存储的主流。一般是SSD和HDD混和使用。
![](https://img-blog.csdnimg.cn/img_convert/4d3bdb5da4460173bc6f04956981ec5b.png)
因为磁头和盘面的的距离很近很近,所以磁盘要防止抖动和防止灰尘。所以磁盘的密封性能一定很好。
1.2.磁盘的存储结构
![](https://img-blog.csdnimg.cn/img_convert/61992ab0cdebb0394f7079ac4b1f9bb1.png)
![](https://img-blog.csdnimg.cn/img_convert/69e1cb990bd70f439becebbd13a16652.png)
1.3.磁盘的逻辑结构
1.3.1.OS对磁盘的方法单位
![](https://img-blog.csdnimg.cn/img_convert/b884f23dc1457b59a7434ba823f3a59b.png)
![](https://img-blog.csdnimg.cn/img_convert/acd194abbe694e9e7d1a76cf91eefccc.png)
![](https://img-blog.csdnimg.cn/img_convert/52c112c47c56951d05c3ef588d72c607.png)
1.3.2.OS对磁盘IO过程的优化
![](https://img-blog.csdnimg.cn/img_convert/eadb9c727b03245caa8234e176108f00.png)
可以stat “文件名”:查看文件具体信息
![](https://img-blog.csdnimg.cn/img_convert/6f1c617dec7112ec5c1e269896142a02.png)
文件系统
2.1.前提知识
![](https://img-blog.csdnimg.cn/img_convert/1a5735db68f5d6f08c4cc1c3602f06ab.png)
![](https://img-blog.csdnimg.cn/img_convert/136baa37921284f9a8872c35135ee6fa.png)
对文件属性和文件内容的认识:
![](https://img-blog.csdnimg.cn/img_convert/1f38e6e5cce755a8a3c9d7f0b81b6e26.png)
2.2.Linux文件系统
![](https://img-blog.csdnimg.cn/img_convert/93add38bd42237e25ef38c65252bcaab.png)
Linux ext2文件系统,上图为磁盘文件系统图(内核内存映像肯定有所不同),磁盘是典型的块设备,硬盘分区被划分为一个个的block。一个block的大小是由格式化的时候确定的,并且不可以更改。例如mke2fs的-b选项可以设定block大小为1024、2048或4096字节。而上图中启动块(Boot Block)的大小是确定的。
Block Group:ext2文件系统会根据分区的大小划分为数个Block Group。而每个Block Group都有着相同的结构组成。政府管理各区的例子
超级块(Super Block):存放文件系统本身的结构信息。记录的信息主要有:bolck 和 inode的总量,未使用的block和inode的数量,一个block和inode的大小,最近一次挂载的时间,最近一次写入数据的时间,最近一次检验磁盘的时间等其他文件系统的相关信息。Super Block的信息被破坏(一般有备份),可以说整个文件系统结构就被破坏了。
GDT,Group Descriptor Table:块组描述符,描述块组属性信息,有兴趣的同学可以在了解一下
块位图(Block Bitmap):Block Bitmap中记录着Data Block中哪个数据块已经被占用,哪个数据块没有被占用
inode位图(inode Bitmap):每个bit表示一个inode是否空闲可用。
inode节点表(inode table):存放文件属性 如 文件大小,所有者,最近修改时间等
数据区(data blocks):存放文件内容
磁盘格式化就是重新写入文件系统。
2.3.读取工作过程
![](https://img-blog.csdnimg.cn/img_convert/214b5c80749fa326058ef616f5112f68.png)
1.先获得文件inode--》 2.通过inode信息获得数据块的下标---》3.完成读取操作
2.4.目录文件
![](https://img-blog.csdnimg.cn/img_convert/e1297fd2421d09f024965ad60208b8ca.png)
2.5.添加文件
1. 存储属性
内核先找到一个空闲的i节点(这里是263466)。内核把文件信息记录到其中。
2. 存储数据
该文件需要存储在n个数据块,内核找到了n个空闲块:300,500,800 --- 。将内核缓冲区的第一块数据复制到300,下一块复制到500,以此类推。
3. 记录分配情况
文件内容按顺序300,500,800存放。内核在inode上的磁盘分布区记录了上述块列表。
4. 添加文件名到目录(在目录中建立文件名和文件inode的映射关系)
新的文件名test.txt。内核将入口(263466,test.txt)添加到目录文件。文件名和inode之间的对应关系将文件名和文件的内容及属性连接起来。
2.6.删除文件
删除文件很简单,只需要修改GDT(块组描述符)。数据块对应的位图结构(block bitmap)和inode对应的位图结构(inode bitmap)即可。其实不是真正意义删除,只是把原有的数据(inode和数据块)从开始的不可覆盖,变成了覆盖,不受保护了。
所以按理说我们删除文件后,知道删除文件的inode (通过日志信息)。只要那一块内存没有被覆盖 是有可能恢复的。
硬链接 和 软链接
3.1.硬链接
介绍
我们看到,真正找到磁盘上文件的并不是文件名,而是inode。其实在linux中可以让多个文件名对应于同一个inode。
touch test.txt :创建一个文件
ln test.txt hard_test.txt :给test.txt创建一个硬链接
ls -1i:查看文件对应的inode
![](https://img-blog.csdnimg.cn/img_convert/7d6cb785c232d1ff037a848332ccaf63.png)
这不就是给inode起一个别名嘛。
用法
![](https://img-blog.csdnimg.cn/img_convert/08d4414374a0fd8701d67ed8e1e84180.png)
也可以看看文件的inode
![](https://img-blog.csdnimg.cn/img_convert/c640927f26fb6160e82dd156dcc3f218.png)
所以每一个目录下的 "." 和 ".." 就是硬链接。
这也就解释了为什么 'cd ..' 能回到上级目录。
特例
"." 和 ".." 就是硬链接,为什么我们不能给目录建立硬链接?
![](https://img-blog.csdnimg.cn/img_convert/d8c89214acecd058ecdfb587884b2a4a.png)
假如我们给根目录建立一个硬连接,当我们从根目录去查找某一个文件的时候,可能就会生成一个闭环,死循环了。
3.2.软链接
介绍
硬链接是通过inode引用另外一个文件,软链接是通过名字引用另外一个文件,所以软连接标定一个文件不是通过文件的inode编号,而是通过文件名称。
ln -s test.txt soft_test.link :给test.txt创建软链接
![](https://img-blog.csdnimg.cn/img_convert/61a4fe3a91c5e00580677c6b30910886.png)
![](https://img-blog.csdnimg.cn/img_convert/51fbba44d81489bdd1a85d6dc9335b4e.png)
这不就是win下的快捷方式嘛。
用法
指向别的可执行程序。(快捷方式)
![](https://img-blog.csdnimg.cn/img_convert/fd0c994159fbd5d42b171e07ff8b1790.png)
![](https://img-blog.csdnimg.cn/img_convert/e51b54285afccd603b77afd348e10b70.png)
3.3.ACM时间
![](https://img-blog.csdnimg.cn/img_convert/ce26296fd324a926289d3ef66ac49f7b.png)
动态库和静态库
4.1.特征介绍
静态库(.a):程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再需要静态库
动态库(.so):程序在运行的时候才去链接动态库的代码,多个程序共享使用库的代码。
一个与动态库链接的可执行文件仅仅包含它用到的函数入口地址的一个表,而不是外部函数所在目标文件的整个机器码在可执行文件开始运行以前,外部函数的机器码由操作系统从磁盘上的该动态库中复制到内存中,这个过程称为动态链接(dynamic linking)
动态库可以在多个程序间共享,所以动态链接使得可执行文件更小,节省了磁盘空间。操作系统采用虚拟内存机制允许物理内存中的一份动态库被要用到该库的所有进程共用,节省了内存和磁盘空间
先来一段测试代码:
![](https://img-blog.csdnimg.cn/img_convert/922eca4e1b85663372497b8154fd1034.png)
一起我们写项目的时候,都会这样使用,在main.c包含各自运用到函数的头文件,然后全部的“xxx.c"文件一起编译,即可编译完成。
![](https://img-blog.csdnimg.cn/img_convert/0379a793dc8f5bbb2b59f6dad11d1bdb.png)
这是我们的以前学到的,头文件和源文件基本使用。、
也可以分开编译最后再链接
![](https://img-blog.csdnimg.cn/img_convert/46b10ee153f6c68d68a82f320aacbd2c.png)
![](https://img-blog.csdnimg.cn/img_convert/b7555ea6e46d9855534b0eb243fc0dc1.png)
可以看到一样可以形成可执行文件。
![](https://img-blog.csdnimg.cn/img_convert/d116c0ad70360e496775c42464561940.png)
![](https://img-blog.csdnimg.cn/img_convert/b29f5acb2252f02ad32cc6abbaf6c19a.png)
所以我们只给别人提供可重定向目标二进制文件和对应的头文件,就可以给别人提供我们的函数使用。
这种思想就已经是库的思想了。
将所以的“xxx.o”文件打包,形成库,提供给对方一个库文件和全部的头文件即可。
打包就是将多个“xxx.o”文件合起来形成一个文件,这就是库文件。
根据打包方法的不同,分为静态库和动态库。
4.2.生成静态库
ar -rc 【生成的库名称】 【全部的“xxx.o”文件】
//ar是gnu归档工具,rc表示(replace and create)(替换并创建)
直接创建一个makefile自动化文件。
![](https://img-blog.csdnimg.cn/img_convert/bef33792ab39a4ec05e2312ccebd95b4.png)
![](https://img-blog.csdnimg.cn/img_convert/a244b38998c443b118252ba404fa5f01.png)
![](https://img-blog.csdnimg.cn/img_convert/8764200991e9e2137f382fac691c1ade.png)
查看生成静态库:
ar -tv 【库文件】:查看库文件是由那俩个文件
t:列出静态库中的文件
v:verbose 详细信息
![](https://img-blog.csdnimg.cn/img_convert/78622cf118bf1b24e908818430624966.png)
![](https://img-blog.csdnimg.cn/img_convert/0e76e8613f897197fcead98c89daa342.png)
交付库的时候:
1.提供:“xxx.a”/“xxx.so”文件;
2.提供对应匹配的 “xxx.h”文件。
一般都会打包好到一个目录,然后再压缩给别人。
4.3.使用静态库
![](https://img-blog.csdnimg.cn/img_convert/6c69c5b55f2bc5532b06019fed5509d3.png)
安装的本质就是:把文件拷贝到知道系统能找到的目录下。
方法一:直接使用
![](https://img-blog.csdnimg.cn/img_convert/78aba28936cd0a7742ea68c07f8ee4dc.png)
gcc的链接规则:
![](https://img-blog.csdnimg.cn/img_convert/99bfdf1258ee00facffced7af2b13db6.png)
1,gcc默认是动态链接(建议行为),对于特定的库,究竟是动态还是静态,取决于你提供的是动态库还是静态库。有动态库就动态链接。
2,有多个库的时候,一个一个链接的,可能一部分是动态链接,一部分是静态链接,只要有一个库是动态链接的,那里就会显示是动态链接。
方法二:安装到系统对应路劲下
![](https://img-blog.csdnimg.cn/img_convert/bee672e88ea114fa0fea2d7a2286eb3f.png)
建议测试完后立即卸载掉,我们自己写的程序放在系统的路劲下不安全。
卸载:就是把刚刚拷贝到系统默认路劲下的文件原封不动的删除掉。一定注意不要删错了。
4.4.生成动态库
![](https://img-blog.csdnimg.cn/img_convert/de8a4b546b056e86cc9f11f2131c59b8.png)
交付库的时候:
1.提供:“xxx.a”/“xxx.so”文件;
2.提供对应匹配的 “xxx.h”文件。
一般会打包好到一个目录下,然后压缩好给别人
![](https://img-blog.csdnimg.cn/img_convert/4554fe67bff98743d5da93dfc762f0fe.png)
4.5.使用动态库
不能像静态链接那样世界使用,运行会找不到对应的动态库。
![](https://img-blog.csdnimg.cn/img_convert/651f1027d809d26606bd7a39df34d394.png)
库的默认搜索路劲
1, 环境变量LD_LIBRARY_PATH
2, /usr/lib/(32位)或/usr/lib64/(64位)
3, /usr/local/lib/(32位) 或 /usr/local/lib64/(64位) (这个是用户级别的)
4, /lib64/
方法一:修改LD_LIBRARY_PATH环境变量
![](https://img-blog.csdnimg.cn/img_convert/be5fd697f2dbf95959b5fcdf433b085b.png)
但是环境变量是内存级的,当我们重新链接我们的机器的时候他就不能再次执行了。因为环境变量改变了。如果像使环境变量不变就需要去修改配置文件了,比较麻烦。临时做测试可以。
方法二:安装进系统
![](https://img-blog.csdnimg.cn/img_convert/5425e2c84cbe8e7a68ea8e8e8b798a17.png)
方法三:配置文件(xxx.conf)
在 /etc/ld.so.conf.d/ 路劲下有很多的配置文件
![](https://img-blog.csdnimg.cn/img_convert/06786856a2bb8c78e3670f97534a26d0.png)
动态库在进行搜索时,包含路劲的配置文件。
创建一个任意名称的xxx.conf 文件
![](https://img-blog.csdnimg.cn/img_convert/00ea03103e4016c39332a3bd8aad6f07.png)
给刚刚创建的配置文件写入,我们的动态库所在的目录(sudo添加)
![](https://img-blog.csdnimg.cn/img_convert/c1ec214198c4a64be1f9424c3bd401f3.png)
然后更新一下:sudo ldconfig(一定要做)
![](https://img-blog.csdnimg.cn/img_convert/c8dbf5c738d6e99edfb2ae0df87bb0f0.png)
这个和安装到系统一样的效果,再次链接还能使用。
删除配置文件的时候也要ldconfig
方法四:创建软连接
在当前目录下建立软连接
![](https://img-blog.csdnimg.cn/img_convert/c690268b4473fd1b08edb5764e4bbfdc.png)
在系统目录下建立软连接
![](https://img-blog.csdnimg.cn/img_convert/1541b01c5575d5cc011f602f6ce1120c.png)
总结
我们能让可执行找到动态库文件的时候,我们再把头文件拷贝进系统,这样就只需指定 ”-l 库名“,即可完成编译任务 。运行时候不需要库文件,但是需要用到动态库(而且还要能找到)。
![](https://img-blog.csdnimg.cn/img_convert/9edc030b9df1b931ed32d4e027924352.png)
动静态库的加载
5.1.静态库
静态库一般不考虑加载过程
因为静态链接编译的时候,就已经将对应的实现加载到了可执行程序你内部,也就是说静态库,在编译完成后,就不需要库了。可执行程序需要的代码,已经被拷贝进去了。
会有代码冗余的现象:例如:每次调用printf,都需要加载一份priintf的实现。
编译是加载,到哪里呢?加载进了代码区。(建议复习进程地址空间)
拷贝进来的代码,必须通过相对确定的地址位置来进行访问。
拷贝进代码区,这里也就和我写的函数一样。采用绝对编址的方式。
5.2.动态库
静态链接:编译的时候在调用函数的位置完成实现代码的拷贝。(代码冗余)
动态链接:编译的时候在调用函数位置放入偏移量地址。
动态库是在使用的时候被加载进内存,然后映射到进程的共享区,确定start地址,然后在可执行程序中存放的是偏移量地址。start地址+偏移量地址确定虚拟地址,通过页表映射完成执行。
![](https://img-blog.csdnimg.cn/img_convert/7afb20fd27fcc2d43cef3d74b207361d.png)