Linux文件系统以及动静态库

news2025/1/12 20:43:43

目录

一、系统 I/O

1.1 接口介绍

1.2 系统调用和库函数

1.3 文件描述符

1.4 重定向

二、理解文件系统

2.1 inode

2.2 硬链接

2.3 软连接

三、动静态库

3.1 初识动静态库

3.2 静态库的打包与使用

3.2.1 打包

3.2.2 使用

3.3 动态库的打包与使用

3.3.1 打包

3.3.2 使用


一、系统 I/O

        对于文件的操作,之前的文章中有写有关于C的接口,其实我们还可以用系统接口进行文件访问,下面我们一起来学习。

1.1 接口介绍

open

pathname:要打开或创建的目标文件

flags:此处可以传入多个常量进行或运算,如下

  • O_RDONLY: 只读打开
  • O_WRONLY: 只写打开
  • O_RDWR : 读,写打开

上面三个常量,必须指定一个且只能指定一个

  • O_CREAT : 若文件不存在,则创建它。需要使用mode选项,来指明新文件的访问权限
  • O_APPEND: 追加写

mode:文件的权限

返回值:成功时返回新打开的文件描述符(下面会讲),失败时返回-1

        类比C接口,系统接口还包括write、read、close、lseek,其使用方式传参与open一致,就不详细介绍了。

1.2 系统调用和库函数

fopen fclose fread fwrite 都是C标准库当中的函数,我们称之为库函数(libc)

而刚刚介绍的 open close read write lseek 都属于系统提供的接口,称之为系统调用接口

我们来看一张操作系统的图:

从上图中可以明显发现,f#系列的函数,都是对系统调用的封装,方便二次开发

1.3 文件描述符

        文件描述符实质就是一个小的整数

        当我们打开文件时,操作系统在内存中要创建相应的数据结构来描述目标文件。于是就有了file结构体,表示一个已经打开的文件对象。而进程执行open系统调用,所以必须让进程和文件关联起来。每个进程都有一个指针*files, 指向一张表files_struct,该表最重要的部分就是包含一个指针数组,每个元素都是一个指向打开文件的指针。所以,本质上文件描述符就是该数组的下标。所以,只要拿着文件描述符,就可以找到对应的文件。

下面我们来做一个实验:连续打开五个文件,看看这五个文件分配的文件描述符分别是多少:

#include<stdio.h>
#include<string.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>

int main()
{
    int fd1 = open("log.txt1",O_RDONLY | O_CREAT,0666);
    int fd2 = open("log.txt2",O_RDONLY | O_CREAT,0666);
    int fd3 = open("log.txt3",O_RDONLY | O_CREAT,0666);
    int fd4 = open("log.txt4",O_RDONLY | O_CREAT,0666);
    int fd5 = open("log.txt5",O_RDONLY | O_CREAT,0666);
    
    printf("fd1: %d\n",fd1);
    printf("fd2: %d\n",fd2);
    printf("fd3: %d\n",fd3);
    printf("fd4: %d\n",fd4);
    printf("fd5: %d\n",fd5);
    
    return 0;
}

运行后发现打开的这几个文件的文件描述符是从3开始的一串连续数字,为什么是从3开始呢?

        因为上面之前说过,创建一个进程时,会默认打开标准输入、标准输出、标准错误,fd_array给他们三个分配的文件描述符分别是0、1、2,因此我们后面创建文件的时候分配的描述符会从3开始。      

如果我们使用close关闭0,1,2其中的一个会怎么样呢?

#include<stdio.h>
#include<string.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>

int main()
{
    close(0);
    int fd = open("log.txt",O_RDONLY | O_CREAT,0666);
    
    printf("fd: %d\n",fd);
    return 0;
}

  

可以发现当我们关闭0即标准输入时,分配给文件的文件描述符为1。

其实文件描述符的分配规则是:从fd_array中找一个最小的、未被使用的,作为这个文件的文件描述符(fd)

1.4 重定向

再来看段代码:

#include<stdio.h>
#include<string.h>
int main()
{
   const char* msg0="hello printf\n";
   const char* msg1="hello fwrite\n";
   const char* msg2="hello write\n";
   printf("%s",msg0);
   fwrite(msg1,strlen(msg1),1,stdout);
   write(1,msg2,strlen(msg2));
   fork();
   return 0;                                                                                                                                                        
}                                                                                                                        
     

我们再对输出进行重定向到file文件中:

        我们发现printf 和fwrite(库函数)都输出了2次,而write 只输出了一次 (系统调用) 。为什么呢? 肯定和fork有关! 

  • 一般C库函数写入文件时是全缓冲的,而写入显示器是行缓冲
  • printf 和fwrite库函数会自带缓冲区(进度条例子就可以说明),当发生重定向到普通文件时,数据的缓冲方式由行缓冲变成了全缓冲
  • 而我们放在缓冲区中的数据,甚至fork之后都不会被立即刷新,只有当进程退出之后,会统一刷新,写入文件当中
  • 但是fork的时候,父子数据会发生写时拷贝,所以当你父进程准备刷新的时候,子进程也就有了同样的一份数据,随即产生两份数据
  • write没有变化,说明没有所谓的缓冲
         printf 和fwrite  是库函数, write 是系统调用,库函数在系统调用的"上层",是对系统调用的"封装”,但是 write 没有缓冲区,而   printf 和fwrite  有,足以说明,该缓冲区是二次加上的,又因为是c,所以由C标准库提供。

二、理解文件系统

2.1 inode

inode(index node)是文件系统中的一个概念,用于存储和管理文件的元数据,包括文件的权限、大小、时间戳等信息,以及文件数据在磁盘上的物理位置。

  • 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是否空闲可用。
  • i节点表:存放文件属性如文件大小,所有者,最近修改时间等
  • 数据区:存放文件内容

我们使用ls -l 命令时,还可以得到除了文件名外的元数据,其中一共包括7种数据,分别是:

模式、硬链接数、文件所有者、文件所属组、大小、最后修改时间和文件名

2.2 硬链接

        其实真正找到磁盘上文件的并不是文件名,而是inode。而在linux中可以让多个文件名对应于同一个inode

        硬链接它不是一个独立的文件,而是一个文件名和inode编号的映射关系,因为它没有自己的inode,我们可以将它理解成源文件的别名

我们可以使用下面的指令来创建一个硬链接:

ln 源文件名称 硬链接文件名称

         使用ls -li 显示它们的 inode 号码和其他相关信息可以看到,硬链接文件和源文件的inode是一样的。

2.3 软连接

        硬链接是通过inode引用另外一个文件,软链接是通过名字引用另外一个文件。

        软链接它有自己独立的inode,是一个独立的文件,有自己的inode属性也有自己的数据块(保存的是指向文件的所在路径+文件名)。

我们可以使用下面的指令来创建一个软链接:

ln -s 源文件名称 硬链接文件名称

        可以看到我们的软链接的inode编号与源文件的inode编号是不一样的,并且软链接的文件大小要比源文件的文件大小要小很多,再来运行一下试试:

        我们可以看到这两个可执行程序运行起来之后打印的结果是一样的。但是软链接的文件大小却要比源文件的大小小很多,这是不是有一种熟悉的感觉呢?

        软链接就相当于是windows下的快捷方式。软链接它的数据块保存的是指向文件的所在路径+文件名,所以我们执行软链接的时候就相当于间接执行了这个源文件。

        但是快捷方式它是不能够单独存在的,当我们的源文件被删除后,尽管有文件名,但是这个软链接就不能再执行了:

三、动静态库

3.1 初识动静态库

我们写一段代码:

#include<stdio.h>

int main()
{
    printf("hello world\n");
    return 0;
}

然后使用ldd指令来查看一个可执行程序所依赖的库文件:

这里的libc.so.6就是当前可执行程序所依赖的库文件,我们通过 ls命令 查看后发现libc.so.6它其实就是一个软链接:

紧接着,我们通过file+文件名来查看一下libc-2.17.so的文件类型:

可以看到libc-2.17.so它其实是一个动态库。

注:

  • 在Linux下,以.so为后缀的文件是动态库,以.a为后缀的文件是静态库
  • 在Windows下,以.dll为后缀的文件是动态库,以.lib为后缀的文件是静态库

在Linux下,我们如何知道一个动静态库的名字呢?

比如libc.so,我们去掉前缀lib,去掉后缀.so,剩下的就是该动静态库的库名了。

在Linux下,gcc/g++编译器默认采用的是动态链接,若是想采用静态链接的话,我们需要在gcc/g++后面带上一个-static选项即可。

  • 静态库(.a):程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再需要静态库.
  • 动态库(.so):程序在运行的时候才去链接动态库的代码多个程序共享使用库的代码
  • 一个与动态库链接的可执行文件仅仅包含它用到的函数入口地址的一个表,而不是外部函数所在目标文件的整个机器码。
  • 在可执行文件开始运行以前,外部函数的机器码由操作系统从磁盘上的该动态库中复制到内存中,这个过程称为动态链接(dynamic linking)。
  • 动态库可以在多个程序间共享,所以动态链接使得可执行文件更小,节省了磁盘空间。操作系统采用虚拟内存机制允许物理内存中的一份动态库被要用到该库的所有进程共用,节省了内存和磁盘空间。

总的来说,动静态库的优缺点如下:

   动态库

优点:

  • 使用动态库生成的可执行程序体积比较小,节省资源。当多个可执行程序同时运行时,若是要使用同一个库,库文件会通过进程地址空间进行共享,内存中不会存在重复代码。

缺点:

  • 比较依赖于第三方库。

   静态库

优点:

  • 使用静态库生成的可执行程序,它不需要依赖于第三方库,即使没有第三方库它也可以独立运行,可移植性高。

缺点:

  • 生成的可执行程序体积比较大,比较占资源。 当多个静态链接生成的可执行程序使用同一个库时,内存中会存在大量重复的代码。比如说:9个静态链接生成的可执行程序都要使用C标准库,那么当这些可执行程序同时运行的时候,内存中就会存在9份C标准库的代码。

3.2 静态库的打包与使用

3.2.1 打包

静态库打包的本质:把代码生成的二进制.o文件进行打包。

接下来举个例子帮助大家深刻了解:

首先创建四个文件,其中两个源文件myadd.c和mysub.c,两个头文件myadd.h和mysub.h:

myadd.h:

#pragma once
#include<stdio.h>

int myadd(int a,int b);

myadd.c:

#include"myadd.h"

int myadd(int a,int b)
{
    return a+b;
}

mysub.h:

#pragma once
#include<stdio.h>

int mysub(int a,int b);

mysub.c:

#include"mysub.h"

int mysub(int a,int b)
{
    return a-b;
}

接下来,根据静态库打包的本质,首先将所有的源文件变成对应的.o目标文件:

然后,使用ar命令将所有的目标文件打包生成静态库:

ar是gnu归档工具,rc表示(replace and create)
我们还可以使用ar命令的-t选项和-v选项来查看静态库中的目录列表:
  • -t:列出静态库重点文件
  • -v(verbose):详细信息

最后,我们将生成的静态库文件与目标文件对应的头文件组织起来:

当我们将我们自己的库给别人使用的时候,我们需要创建两个目录,一个用来存放所有的头文件一个用来存放静态库文件

因此,我们将myadd.h与mysub.h这两个头文件放到include这个目录下将静态库文件放到lib这个目录下。然后将这两个目录都放到mylib这个目录下,此时我们若是向把我们自己的库给别人使用,我们只需要将mylib这个文件发给别人就可以了

3.2.2 使用

我们首先创建一个mytest.c文件,然后写一段简单的代码用一下加法函数:

#include"myadd.h"
int main()
{
    int x = 10;
    int y = 20;
    printf("add: %d\n",myadd(x,y));
    return 0;
}

因为编译器它并不知道你所包含的myadd.h头文件在哪里,所以我们需要指定头文件的搜索路径。

因此我们需要在makefile里面的gcc后面带一个选项  -I(大写i) 就可以指定头文件的搜索路径了:

make后发现报错:

这是因为头文件myadd.h里面只有加法函数的声明,并没有函数的定义,因此我们还需要指定库文件的搜索路径。

因此我们只需要在makefile里面的gcc后面带一个选项  -L  就可以指定库文件的搜索路径了,与此同时,因为你只是告诉了编译器库文件的搜索路径在哪里,但是你并没有告诉你要去链接哪一个库,假如说这个库文件里面有很多个库,那编译器它又怎么知道你想要去链接哪个库呢。

所以我们还需要指定一下我们需要链接库文件中的哪一个库

这就需要在makefile里面的gcc后面再带一个选项  -l (小写L)去指明我们想要链接库文件中的哪一个库:

注意:前面说过怎么看一个库文件的名字(去掉前缀去掉后缀),因为我们是要告诉编译器我们想要链接库文件中的哪个库,所以我们只需要在-l选项后面带上它的名字即可。

大功告成。

除此之外,我们还可以将头文件和库文件放到系统路径下,来达到使用的目的。

如果我们不指定搜索路径的话,编译器它就找不到我们的头文件和库文件,除了上面指定搜索路径的法子我们还可以将我们的头文件和库文件放到系统路径下,这样的话编译器就能够找到了:

虽然说这种方法相较于第一种方法而言会简单一点,但是我不建议你使用这种方法,原因有两个:

  1. 将我们的头文件和库文件放到系统路径下,这样做有可能会对系统文件造成污染
  2. 使用第一种方法可以帮你熟悉gcc命令,同时使用第一种方法虽然麻烦但是不会对系统文件造成污染。

3.3 动态库的打包与使用

3.3.1 打包

有了静态库的基础,就可以很容易地理解动态库了:

首先,让所有的源文件生成对应的.o目标文件

​​​​​​​

其中有三个我们比较陌生的选项:

  • shared: 表示生成共享库格式
  • fPIC:产生位置无关码
  • < 表示当前规则中的第一个依赖文件(即myadd.c,第六行为mysub.c)

然后,将头文件和我们生成的动态库组织起来,这里我们换一种方式来将它们两个组织起来,我们在Makefile里面编写一段代码,通过 发布 然后将他们给组织起来:

3.3.2 使用

发布成功后,我们继续写一段测试代码吧:

test.c:

#include"myadd.h"
#include"mysub.h"

int main()
{
    int a = 20;
    int b = 10;
    printf("add: %d\n",myadd(a,b));
    printf("sub: %d\n",mysub(a,b));
    
    return 0;
}

结果发现运行时显示报错:

        但我们明明已经指定头文件的搜索路径,指定库文件的搜索路径,以及我们要链接库文件中的哪一个,为什么我们这里还是会报错呢?

        这是因为这只是告诉了编译器,但并没有告诉操作系统,而链接动态库的时候是在程序运行的时候链接的。

因此我们可以使用以下两种方法解决:

1. 拷贝.so文件到系统共享库路径下

2. 更改LD_LIBRARY_PATH

LD_LIBRARY_PATH是程序运行时帮我们找动态库的路径的,因此我们将该动态库所在目录路径添加到LD_LIBRARY这个环境变量中就可以了:

至此,本章的内容就介绍完啦,如果觉得对你有帮助的话可以点赞关注,谢谢啦~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1261592.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

四川劳动保障杂志社四川劳动保障编辑部四川劳动保障杂志2023年第10期目录

主题报道 四川抢抓“金九银十”招聘季多措并举稳就业促就业 举措频“上新” 金秋送岗忙 张玉芳; 2-5 法眼《四川劳动保障》投稿&#xff1a;cnqikantg126.com 筑牢长期护理保险基金安全防线 李科仲;赖晓薇; 6-7 调研 提升职业技能培训工作的举措 寇爵; 8-9 城乡…

2023.11.26 关于 Spring Boot 单元测试

目录 单元测试 优势 单元测试的使用 具体步骤 实现不污染数据库 阅读下面文章之前 建议点击下方链接了解 MyBatis 的创建与使用 MyBatis 的配置与使用 单元测试 单元测试 指对软件中的最小可测试单元进行检查和验证的过程单元测试 由开发人员在编码阶段完成&#xff0c;…

销售人员应该具备哪些良好心态和素养

销售人员应该具备哪些良好心态和素养 作为市场的前线战士&#xff0c;销售人员的心态与素养对于成功销售至关重要。以下所列举的&#xff0c;正是销售人员不可或缺的十种良好心态与专业素养&#xff0c;它们将帮助你更好地应对销售挑战&#xff0c;提升业绩。 1. 积极乐观的态…

11.盛最多的水的容器

一、题目 给定一个长度为 n 的整数数组 height 。有 n 条垂线&#xff0c;第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线&#xff0c;使得它们与 x 轴共同构成的容器可以容纳最多的水。 返回容器可以储存的最大水量。 题目难度&#xff1a;中等 示例&a…

【备忘录】快速回忆ElasticSearch的CRUD

导引——第一条ElasticSearch语句 测试分词器 POST /_analyze {"text":"黑马程序员学习java太棒了","analyzer": "ik_smart" }概念 语法规则 HTTP_METHOD /index/_action/IDHTTP_METHOD 是 HTTP 请求的方法&#xff0c;常见的包括…

Flask 运用Xterm实现交互终端

Xterm是一个基于X Window System的终端仿真器&#xff08;Terminal Emulator&#xff09;。Xterm最初由MIT开发&#xff0c;它允许用户在X Window环境下运行文本终端程序。Xterm提供了一个图形界面终端&#xff0c;使用户能够在图形桌面环境中运行命令行程序。而xterm.js是一个…

物理世界中的等距3D对抗样本

论文题目&#xff1a;Isometric 3D Adversarial Examples in the Physical World 会议&#xff1a;NIPS 2022 点云&#xff1a; 点云——表达目标空间分布和目标表面特性的海量点集合&#xff0c;点包含xyz坐标信息 能够包含颜色等其他信息 使用顶点、边和面的数据表征的三维…

【操作宝典】SQL巨擘:掌握SQL Server Management的终极秘籍!

目录 ⛳️【SQL Server Management】 ⛳️1. 启动登录 ⛳️2. 忘记密码 ⛳️3. 操作数据库和表 3.1 新建数据库text 3.2 新建表 3.3 编辑表 3.4 编写脚本 ⛳️【SQL Server Management】 ⛳️1. 启动登录 需要开启服务 ⛳️2. 忘记密码 登录windows--> 安全性 -->…

基于Java SSM框架+Vue留学生交流互动论坛网站项目【项目源码+论文说明】

基于java的SSM框架Vue实现学生交流互动论坛网站演示 摘要 21世纪的今天&#xff0c;随着社会的不断发展与进步&#xff0c;人们对于信息科学化的认识&#xff0c;已由低层次向高层次发展&#xff0c;由原来的感性认识向理性认识提高&#xff0c;管理工作的重要性已逐渐被人们所…

四川开启智能巡河形式,无人机水利行业应用再创新

在四川省某区域&#xff0c;复亚智能无人机系统以其独特的机场网格化部署得到成功应用&#xff0c;覆盖了该区域内多条市级、省级河流&#xff0c;成为水利行业的新亮点。这一先进系统以无人机水利行业应用为核心&#xff0c;通过网格化和信息化手段&#xff0c;实现了对水域环…

Java核心知识点整理大全19-笔记

目录 14.1.5.2. MemStore 刷盘 全局内存控制 MemStore 达到上限 RegionServer 的 Hlog 数量达到上限 手工触发 关闭 RegionServer 触发 Region 使用 HLOG 恢复完数据后触发 14.1.6.HBase vs Cassandra 15. MongoDB 15.1.1. 概念 15.1.2. 特点 16. Cassandra 16.1.1…

activate jrebel JRebel激活过程

1.下载反向代理 地址&#xff1a;https://pan.baidu.com/s/1wklvDtyrSBXE4I6lKCxXBg?pwdidos 下载完后双击运行 2.在idea中如下步骤点击&#xff1a;File ——> Setting... ——> JRebel ——> Activate now 填入Team URL 1.http://127.0.0.1:8888/uuid //这个…

用于图像分类任务的经典神经网络综述

&#x1f380;个人主页&#xff1a; https://zhangxiaoshu.blog.csdn.net &#x1f4e2;欢迎大家&#xff1a;关注&#x1f50d;点赞&#x1f44d;评论&#x1f4dd;收藏⭐️&#xff0c;如有错误敬请指正! &#x1f495;未来很长&#xff0c;值得我们全力奔赴更美好的生活&…

电脑如何录音?适合初学者的详细教程

“电脑怎么录音呀&#xff1f;参加了一个学校举办的短视频大赛&#xff0c;视频拍摄都很顺利&#xff0c;音乐却出了问题&#xff0c;朋友说可以用电脑录制一段音乐应付一下&#xff0c;可是我不会操作&#xff0c;有哪位大佬教教我&#xff01;” 声音是一种强大的媒介&#…

回归预测 | MATLAB实现SMA+WOA+BOA-LSSVM基于黏菌算法+鲸鱼算法+蝴蝶算法优化LSSVM回归预测

回归预测 | MATLAB实现SMAWOABOA-LSSVM基于黏菌算法鲸鱼算法蝴蝶算法优化LSSVM回归预测 目录 回归预测 | MATLAB实现SMAWOABOA-LSSVM基于黏菌算法鲸鱼算法蝴蝶算法优化LSSVM回归预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 MATLAB实现SMAWOABOA-LSSVM基于黏菌算法…

Ubuntu18.04磁盘取证-中难度篇

涉及的镜像文件&#xff1a; sdb.vhd uac.tar ubuntu.20211208.mem 需要利用的工具&#xff1a; volatility3 volatility2.6.1 FTK/Autopsy Strings 题干 容器是一个Ubuntu Linux 蜜罐&#xff0c;用来观察利用 CVE-2021-41773 的漏洞攻击者想要做什么。 您将看到一个 cr…

高仿IT之家微信小程序(附精选源码32套,涵盖商城团购等)

项目预览 主要包含主页资讯&#xff0c;圈子俩大模块 主页 资讯详情 圈子 相关代码 网络请求 import wx from wx import Fly from flyioconst request new Fly()request.interceptors.request.use((request) > {wx.showNavigationBarLoading()return request })requ…

Linux - 动静态库(下篇)

前言 在上篇博客当中&#xff0c;对静态库是什么&#xff0c;怎么使用&#xff0c;简单实现自己的静态库&#xff0c;这些做了描述&#xff0c;具体请看上篇博客&#xff1a; 本篇博客将会对 动态库是什么&#xff0c;怎么使用&#xff0c;简单实现自己的动态库&#xff0c…

什么是yum?

&#x1f388;个人主页:&#x1f388; :✨✨✨初阶牛✨✨✨ &#x1f43b;推荐专栏1: &#x1f354;&#x1f35f;&#x1f32f;C语言初阶 &#x1f43b;推荐专栏2: &#x1f354;&#x1f35f;&#x1f32f;C语言进阶 &#x1f511;个人信条: &#x1f335;知行合一 &#x1f…

浅谈安科瑞多功能仪表和网关在中国香港某项目的应用

摘要&#xff1a;本文介绍了安科瑞多功能电能表在中国香港某项目的应用。APM系列交流多功能仪表是一款专门为电力系统、工矿企业、公用事业和智能建筑用于电力监控而设计的智能电表。 Abstract&#xff1a;This article introduces the application of the IoT power meter in…