当前位置: 首页 > web >正文

Linux基础IO(十一)之动态库(基础IO的最后一篇啦!)

文章目录

  • 动态库
    • 生成动态库
    • 使用动态库
    • 现象事实
    • 使用外部库
    • 动态库怎么被加载的
    • 进程地址空间的第二讲
      • 关于地址
        • 1.程序没有加载前的地址(程序)
        • 2.程序加载后的地址(进程)
        • 3.动态库的地址

动态库

生成动态库

shared: 表示生成共享库格式

fPIC:产生位置无关码(position independent code)

库名规则:libxxx.so

image-20250405155951292

myprintf.h

#pragma once  
#include <stdio.h>  
void print();  

myprintf.c

#include "myprintf.h"void print()  
{  printf("hello print!\n");  printf("hello print!\n");  printf("hello print!\n");
}

mylog.h

#pragma once
#include <stdio.h>void log(const char*);  

mylog.c

#include "mylog.h"  void log(const char*msg)  
{  printf("hello %s\n",msg);
}

动静态库都要先形成.o文件,gcc -c不指定名字,默认生成同名 .o 文件。

可执行权限x,本质就是该文件是否会以可执行文件的形式加载到内存里。静态库不需要加载。

image-20250405170805770

image-20250405170847292

main.c

#include "myprintf.h"
#include "mylog.h"int main(){print();Log("linux");return 0;                                               }

使用动态库

编译选项

l:链接动态库,只要库名即可(去掉lib以及版本号)
L:链接库所在的路径.

image-20250405171728502

命令行上的路径是告诉编译器的,

动态库的路径也得告诉系统–加载器

加载也需要路径,系统会去特定的路径下寻找

解决加载找不到路径的方法:

  1. 拷贝到系统默认的库路径下 /lib64 /usr/lib64 (最常用)【直接安装到系统里】
  2. 在系统默认的库路径下 /lib64 /usr/lib64 建立软链接

image-20250405173551278

  1. 将自己的库所在的路径,添加到系统的环境变量 LD_LIBRARY_PATH

    重启 XShell 对应配置的环境变量就没了(可以加到环境变量的配置文件中)

image-20250405174324326

  1. /etc/ld.so.conf.d 建立自己动态库路径的配置文件,然后重新 ldconfig 即可

这个方法是永久有效的。重启XShell也会存在。

findway.conf

/home/lll/linux_code/lesson24/test/mylib/lib

image-20250405175543724

现象事实

1.动态库在进程运行的时候是要被加载的。(静态库不用)

2.常见的动态库被所有的(动态链接的)可执行程序都要使用。

动态库–共享库

所以,动态库在系统中加载之后,会被所有进程共享。

main.c

#include "mymath.h"
#include "myprintf.h"
#include "mylog.h"int main()
{int n=div(10,0);printf("10/0=%d,myerror=%d\n",n,myerrno);print();Log("linux");return 0;
}

image-20250405203033076

所需要的代码已经拷贝到可执行程序了,所以静态库删了也不影响程序运行。

image-20250405203232303

动态库删除了 两份代码都跑不了了

main.ctest.c

#include "myprintf.h"
#include "mylog.h"int main()
{print();Log("linux");return 0;
}

image-20250405204828429

一个共享库会被两个不相关的代码使用。

一个共享库挂了,那么和这个共享库相关的代码都跑不了了。

使用外部库

系统中其实有很多库,它们通常由一组互相关联的用来完成某项常见工作的函数构成。

比如用来处理屏幕显示情况的函数(ncurses库)

-lm表示要链接libm.so或者libm.a库文件

动态库怎么被加载的

动态库加载

进程地址空间的第二讲

什么是虚拟地址?什么是物理地址?

CPU读到的指令里面用的是什么地址?

gcc -fPIC --与位置无关码


关于地址

1.程序没有加载前的地址(程序)

程序编译好之后,内部有地址的概念吗? 有!!!

可执行程序分成很多段!(代码段、数据段)

平坦模式–编译器也要考虑操作系统!!!

已经是虚拟地址了!-- 逻辑地址!(段地址+偏移量)

image-20250405221313424

image-20250406204834649

2.程序加载后的地址(进程)

test.c

#include <stdio.h>int a=10;
int b=20;
int main()
{printf("Hello linux\n");int x=30;int y=40;int z=x+y;printf("ret=%d\n",z);return 0;
}

把二进制程序反汇编出来了。

左边一列就是地址。每条指令都有自己的长度。

地址可以不出现。

程序入口+每条指令的长度就可以让代码运行了,

call指令可以计算出偏移量来找到对应的代码。

在运行之前,CPU中的寄存器提前内置了指令集(精简指令集|复杂指令集)

image-20250406203213671

image-20250406203431418

image-20250406213207062

3.动态库的地址

image-20250406214611856

image-20250406221720728

静态库为什么不谈加载?不谈与位置无关?

1.静态库会直接把对应的方法程序拷贝到可执行程序里,所以不需要加载。

2.静态库与位置有关!

因为静态库程序是直接拷贝到可执行文件里的,

可执行文件在程序地址空间中是直接绝对编址,有绝对地址,

所以地址在虚拟地址中是确定的!

http://www.xdnf.cn/news/2094.html

相关文章:

  • 智能座舱背后的秘密:智能座舱测试如何“具身智能”
  • Golang | 自行实现并发安全的Map
  • 在 WSL 安装 OpenFOAM-12
  • 【软件工程】TCP三次握手中的SYN与ACK:核心机制详解
  • 使用ast解ob混淆时关于types的总结
  • leetcode刷题日记——简化路径
  • AI与思维模型【79】——煤气灯效应
  • 深入解析Mlivus Cloud核心架构:rootcoord组件的最佳实践与调优指南
  • 【金仓数据库征文】交通行业的国产化数据库替换之金仓数据库KingbaseES应用实践
  • 【风控】稳定性指标PSI
  • 基于STM32、HAL库的MAX31865模数转换器ADC驱动程序设计
  • 消息队列mq在Mlivus Cloud向量数据库中的关键配置与最佳实践
  • C++智能指针概念理解的面试题
  • window.location.href的用法
  • 基于 Netmiko 的网络设备自动化操作
  • 《逐梦九天:中国航天编年史》
  • QT文本框(QTextEdit)设置内容只可复制粘贴
  • C++:继承机制详解
  • Cursor 配置 MCP Tool
  • 写在后面的话
  • yolo常用操作(长话短说)热力图,特征图,结构图,训练,测试,预测
  • 打开Qt应用程序以控制台
  • Linux基础篇、第四章_02磁盘及分区管理fdisk 和 gdisk
  • 厚铜PCB打样全流程解析:从文件审核到可靠性测试的关键步骤
  • Python的库
  • Hbase集群管理与实践
  • C语言——字串处理
  • 什么是快应用
  • STM32 I2C总线通信协议
  • 遥感金融风险监管:技术革新与实践探索