125 Linux C++ 系统编程4 Linux 静态库制作,动态库制作,静态库和动态库对比。静态库运行时找不到库的bug fix

一 静态库 和动态库 对比

静态库的原理:假设我们有一个 静态库,大小为500M,这个静态库实现了一些打牌的逻辑算法,提供了一堆API,让开发者 可以轻松的实现 54张扑克牌的随机发牌,指定发牌等功能。

我们写了一个腾讯的棋牌类游戏,在牌类中,有斗地主,够级,升级等游戏,很显然,每一个单独的游戏都是一个 可执行的 .out文件了。

假设叫做 a.out,b.out,c.out

显然这a.out,. b.out,c.out都会用到静态库中的这些方法。

那么静态库 和 每一个 .out结合的样子 ,类似下图:

也就是说:

静态库会被每个.out copy 一份到自己的代码里面。

静态库要求执行效率高,但是会牺牲空间。

操作系统的开机加载的一般都是静态库。

目前静态库的应用比较少,动态库的场景比较多,静态库知道怎么弄,如果今后开发中遇到了,知道怎么实做就可以了。

静态库在应用程序生成后,可以不必再编译,节省再编译时间,如果其他开发人员要使用您的程序,而你又不想给其源码,提供静态库是一种选择。

动态库则不会 被copy

动态库会单独的存放一份,被大家共享,不会被各个 .out文件copy 一份。而是在a.out需要的时候去动态库中找到想要的 api,调用一下。

二 如果制作一个静态库

1.使用写好的 .c.h.cpp文件生成.o文件

        

g++ -c addfunc.cpp -I ./head -o addfunc.o
g++ -c subfunc.cpp -I ./head -o subfunc.o
g++ -c mulfunc.cpp -I ./head -o mulfunc.o
g++ -c devfunc.cpp -I ./head -o devfunc.o

2.build 出来静态库文件,我们这里起名叫做 libdou.a

ar rcs libdou.a addfunc.o subfunc.o mulfunc.o devfunc.o

这时候我们就得到了 libdou.a 

关于1,2的说明

在build 出来静态库之前,是需要生成所有.c 和 .cpp的.o文件的,我们这里将.h文件都放在了head头文件里面

我们使用的代码如下: 有5个文件,一个.h 放在 head文件夹下,4个.cpp文件

dou.h
#pragma once
#include <iostream>
using namespace std;
//目前的状况是:我们是第三方的库开发者,致力于开发一些第三方库卖钱,开发了一个加减乘除的算法库,
//这个算法很先进,使用了AI技术,是8个博士后的心血结晶,我们不希望使用者知道开发的细节,
//因此我们需要提供一个 .h文件,告知使用者你要引入的头文件是这个.h文件
//并且提供了一个静态库给 开发者,libdou.aint addfunc01(int a, int b);
int subfunc02(int a, int b);
int mulfunc03(int a, int b);
int devfunc04(int a, int b);

addfunc.cpp
#include "dou.h"int addfunc01(int a, int b) {return a + b;
}

subfunc.cpp
#include "dou.h"int subfunc02(int a, int b) {return a - b;
}

mulfunc.cpp#include "dou.h"int mulfunc03(int a, int b) {return a * b;
}

devfunc.cpp#include "dou.h"int devfunc04(int a, int b) {if (b == 0) {cout << "devfunc04 error because dividend ==0" << endl;return -1;}return a / b;
}

g++ -c addfunc.cpp -I ./head -o addfunc.o
g++ -c subfunc.cpp -I ./head -o subfunc.o
g++ -c mulfunc.cpp -I ./head -o mulfunc.o
g++ -c devfunc.cpp -I ./head -o devfunc.o
ar rcs libdou.a addfunc.o subfunc.o mulfunc.o devfunc.o

这时候我们就得到了 libdou.a 

3. 第三方公司如何应用

3.0我们卖给第三方公司的就是一个头文件dou.h,和一个libdou.a,

3.1 第三方的有一个 test.cpp,在这个 test.cpp 用到了静态库中的一些方法

需要导入我们的 #include "dou.h"

并且将dou.h 放在和test.cpp 一行的 head目录下

test.cpp
#include <iostream>
#include "dou.h"
using namespace std;int main(){cout<<"a+b = " << addfunc01(10,5)<<endl;cout<<"a-b = " << subfunc02(10,5)<<endl;cout<<"a*b = " << mulfunc03(10,5)<<endl;cout<<"a/b = " << devfunc04(10,5)<<endl;return 0;
}

3.2 将test.cpp 和 静态库文件 libdou.a 静态编译在一起。生出来一个test.out文件

g++ test.cpp libdou.a -o test.out -I ./head

3.3执行 ./test.out

hunandede@hunandede-virtual-machine:~/day02/staticlib$ g++ test.cpp libdou.a -o test.out -I ./head
hunandede@hunandede-virtual-machine:~/day02/staticlib$ ./test.out
a+b = 15
a-b = 5
a*b = 50
a/b = 2

3.4 查看 test.out的大小

我们观察到 最终客户生成的 test.out 占据的大小为 14016,但是实际上我们的test.cpp文件只有269这么大。可见确实 将静态库 build 自己里面了。

三 .动态库的制作和使用

 1.使用写好的 .c.h.cpp文件生成.o文件

  但是这种动态库的.o文件,是和静态库的.o文件不一样。因此制作方法也不一样。

动态库制作.o文件的方法:

g++ -c addfunc.cpp -I ./head -o addfunc.o -fPIC

静态库制作.o文件的方法:

g++ -c addfunc.cpp -I ./head -o addfunc.o

g++ -c addfunc.cpp -I ./head -o addfuncdongtai.o -fPIC
g++ -c subfunc.cpp -I ./head -o subfuncdongtai.o -fPIC
g++ -c mulfunc.cpp -I ./head -o mulfuncdongtai.o -fPIC
g++ -c devfunc.cpp -I ./head -o devfuncdongtai.o -fPIC

如下是两者不同原理性的说明:能看懂就看,看不懂拉倒.记住前面的结论就可以了

静态库中方法的地址是以main为依据,一般都是main的地址+xxx 

例如我们的 addfunc01方法的地址 就是 main的地址+100, (注意,这里100不是一个真实的值,是我们猜测的)

在编译的第四阶段,链接的时候,会将main的地址给定一个确定的值,因此我们调用addfunc01的时候,地址也就确定了

动态库的方法的地址不是以main为依据的,只有在调用到 addfunc01dongtai 方法的时候,才去找真实的地址,因此也叫做动态绑定,查看printf函数的反汇编,会有 <printf@plt>的字样,知道这里就可以了。

2,。使用 g++ -shared 制作动态库

gcc -shared -o lib库名.so add.o sub.o div.o

使用参数 -shard 

-o 重命名 

lib库名.so  为我们要制作出来的 动态库文件

g++ -shared -o libdoudongtai.so addfuncdongtai.o subfuncdongtai.o mulfuncdongtai.o devfuncdongtai.o 

3.编译可执行程序时,指定所使用的动态库,-l :指定库名 -L:指定库路径

g++ test.cpp -o a.out -lmymath -L./lib

注意我们实现的时候并没有将libdoudongtai.so放在 lib目录下,而是和test.cpp放在一起了,只是将将.h文件放在 head文件夹下。

 g++ test.cpp -o a.out -ldoudongtai -L./ -I./head

4.运行 可执行程序 发生问题

./a.out

./a.out: error while loading shared libraries: libdoudongtai.so: cannot open shared object file: No such file or directory
 

5.动态库 运行原理 以及 bug fix

原因:动态库想要执行需要两个关键的地方:链接器 和 动态链接器

链接器 和 动态链接器没有关系,可以理解为 “张三” 和 “张三丰” 的关系。

        链接器 :工作于编译的 链接阶段,工作时需要 -l 和 -L 支持,我们已经在前面说明了在哪里

        动态链接器:工作于程序运行阶段,工作时需要提供动态库所在目录位置。

                                动态库所在目录位置是几个比较固定的位置,我们需要将 动态库所在目录 放在这几个固定的位置,如下的说明,应该是有三个地方

​
当执行函数动态链接.so时,如果此文件不在缺省目录下‘/lib’ and ‘/usr/lib’.那么就需要指定环境变量LD_LIBRARY_PATH​

       1.动态环境变量存放的地方:为 LD_LIBRARY_PATH

                export 表示导入,下面的意思是,导入LD_LIBRARY_PATH为当前文件夹。

                export LD_LIBRARY_PATH=./

                然后执行 ./a.out     结果正常

        2. 当我们将当前的 窗口关闭,然后重新打开一个窗口的时候,cd到当前目录,执行 ./a.out 又无法执行了, 这是因为 LD_LIBRARY_PATH 环境变量是 进程的概念,我们刚才打开的 窗口中执行了 export LD_LIBRARY_PATH=./  当这个窗口关闭后,设置的也就不生效了。

        那么怎么弄呢? 既然窗口关闭后,LD_LIBRARY_PATH 环境变量的值就会使用默认的,那么我们就改动 窗口的配置即可。从前几章的知识我们知道 ,我们在窗口打的字,最终是 shell 解释器在处理内容,然后回复给我们,shell有很多种,而在unbutu中,这个shell 实际上 bash。因此我们改动bash的配置文件就OK了。bash的配置文件叫做 .bashrc   

打开 .bashrc ,添加 export LD_LIBRARY_PATH=./

注意后面的./ 是路径,如果我们这么加,每次都需要进入到 a.out的目录才能执行a.out

建议 改成绝对路径, 打开 .bashrc ,添加 export LD_LIBRARY_PATH=/home/hunandede/day02/dongtaiku/

总结

上面写的太多了。整理

5.1 临时生效方法,只在当前窗口有用

export 表示导入,下面的意思是,导入LD_LIBRARY_PATH为当前文件夹。

                export LD_LIBRARY_PATH=./

        

5.2 永久生效的方法:配置bash的 配置文件 .bashrc

1.打开终端,vim ~/.bashrc

2.在最后一行加上

export LD_LIBRARY_PATH=/home/hunandede/day02/dongtailib

保存,退出

3 执行  . .bashrc/ 重启终端  (这个好像不行,先不管)

或者 source .bashrc  重启终端

或者关闭终端后,重新打开

4.执行 ./a.out

6.动态库bug fix2 ,加入到 ‘/lib’ 或者‘/usr/lib’ 中

当执行函数动态链接.so时,如果此文件不在缺省目录下‘/lib’ and ‘/usr/lib’.

才会去找 LD_LIBRARY_PATH

因此我们也可以将 .so文件copy 一份 放在 根目录下的 /lib文件下。

7.动态库 bug fix3,配置文件方法-- 修改etc/ld.so.conf

1.修改etc/ld.so.conf

sudo vim /etc/ld.so.conf

添加你的共享库路径

2.更新查找共享库的路径 -v 是显示个用户看过程的意思

sudo ldconfig -v

3.测试你的程序可否找到共享库

ldd a.out

8.怎么知道 a.out 文件是否已经有所有的动态库了呢?

可以使用 ldd  a.out 查看

ldd 是这个命令,它会分析 a.out执行起来需要哪些动态库,以及这些动态库执行起来的路径在哪里,如果你的动态库缺失,或者没有配置,则后面为空

如下:我们的 a.out是OK的,因此查看

hunandede@hunandede-virtual-machine:~/day02/dongtailib$ ldd a.outlinux-vdso.so.1 =>  (0x00007ffe28528000)libdoudongtai.so => /home/hunandede/day02/dongtailib/libdoudongtai.so (0x00007fb421ae1000)libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007fb42175f000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fb421395000)libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007fb42108c000)/lib64/ld-linux-x86-64.so.2 (0x00007fb421ce4000)libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007fb420e76000)

失败的样子

9.那么这里又有一个问题了,当我们给了user  .h文件和  .so文件后,还要教客户怎么在 配置吗?

实际开发中怎么做的呢?

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

自定义悬浮气泡组件

一.常用悬浮气泡展示 在一个项目中&#xff0c;常常会使用点悬浮展示&#xff0c;而市面上悬浮tooltip的组件非常多 例如常用的antd提供的Tooltip 用法如下&#xff08;来自于官方文档示例&#xff09;&#xff1a; import React from react; import { Button, Tooltip, Con…

第一讲《八大人觉经》

《八大人觉经》。诸位法师。诸位居士。阿弥陀佛。好&#xff0c;大家请放掌。 我们这次&#xff0c;大家有很殊胜的因缘&#xff0c;由于净土梵音的启请&#xff0c;大家能够暂时的放下人世的尘劳&#xff0c;来共同学习《八大人觉经》。在讲述本经之前&#xff0c;我想简单地…

mysql-MVCC

一、基础概念 1. MVCC的含义 MVCC (Multiversion Concurrency Control)&#xff0c;即多版本并发控制技术&#xff0c;它是通过读取某个时间点的快照数据&#xff0c; 来降低并发事务冲突而引起的锁等待&#xff0c; 从而提高并发性能的一种机制. MVCC 的实现,是通过保存数据…

【无刷电机学习】各种电机优势比较

目录 0 参考出处 1 有刷与无刷比较 2 交流与直流比较 3 内转子与外转子比较 4 Delta型与Y型定子绕向比较 5 低压BLDC的一些优点 0 参考出处 【仅作自学记录&#xff0c;不出于任何商业目的。如有侵权&#xff0c;请联系删除&#xff0c;谢谢&#xff01;】 维基百科…

动态规划(算法竞赛)--线性DP编辑距离

1、B站视频链接&#xff1a;E07 线性DP 编辑距离_哔哩哔哩_bilibili 题目链接&#xff1a;编辑距离 - 洛谷 #include <bits/stdc.h> using namespace std; char a[2010],b[2010]; int f[2010][2010]; int main(){scanf("%s %s",a,b);int mstrlen(a),nstrlen(b…

C++笔记:二叉搜索树(Binary Search Tree)

文章目录 二叉搜索树的概念二叉搜索树操作1. 框架搭建2. 遍历3. 查找迭代实现递归实现 4. 插入迭代实现递归实现 5. 删除迭代实现递归实现 6. 析构与销毁7. 拷贝构造与赋值重载 二叉搜索树的应用二叉搜索树的性能分析二叉搜索树模拟实现源码 二叉搜索树的概念 二叉搜索树又称二…

HL祭记汇

一.写在前面 如果说廿四10天集训&#xff0c;对于我&#xff0c;是完成了从入门&#xff08;虽然可能我比别人入门更早&#xff1f;&#xff09;到准OIer的蜕变&#xff0c;那么&#xff0c;HL7天&#xff0c;可以说是真正成为了OIer&#xff0c;虽然是被小学生、初中生&#…

pclpy KD-Tree K近邻搜索

pclpy KD-Tree K近邻搜索 一、算法原理1.KD-Tree 介绍2.原理 二、代码三、结果1.原点云2.k近邻点搜索后的点云 四、相关数据 一、算法原理 1.KD-Tree 介绍 kd 树或 k 维树是计算机科学中使用的一种数据结构&#xff0c;用于在具有 k 维的空间中组织一定数量的点。它是一个二叉…

【Flink精讲】Flink组件通信

主要指三个进程中的通讯 CliFrontendYarnJobClusterEntrypointTaskExecutorRunner Flink内部节点之间的通讯使用Akka&#xff0c;比如JobManager和TaskManager之间。而operator之间的数据传输是利用Netty。 RPC是统称&#xff0c;Akka&#xff0c;Netty是实现 Akka与Ac…

【Vue3】学习computed计算属性

&#x1f497;&#x1f497;&#x1f497;欢迎来到我的博客&#xff0c;你将找到有关如何使用技术解决问题的文章&#xff0c;也会找到某个技术的学习路线。无论你是何种职业&#xff0c;我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章&#xff0c;也欢…

python 提取PDF文字

使用pdfplumber&#xff0c;不能提取扫描的pdf和插入的图片。 import pdfplumberfile_path rD:\UserData\admindesktop\官方文档\1903_Mesh-Models-Overview_FINAL.pdf with pdfplumber.open(file_path) as pdf:page pdf.pages[0]print(page.extract_text()) # 所以文字prin…

【c++设计模式04】创建型2:工厂方法模式(Factory Pattern)

【c设计模式04】创建型2&#xff1a;工厂方法模式&#xff08;Factory Pattern&#xff09; 一、工厂模式二、简单工厂模式的弊端三、工厂方法模式四、UML类图五、demo六、总结 原创作者&#xff1a;郑同学的笔记 原创地址&#xff1a;https://zhengjunxue.blog.csdn.net/artic…