Linux动静态库

在这里插入图片描述

文章目录

    • 1. 静态库
    • 2. 动态库
    • 3. 动态库的加载

本章代码gitee仓库:动静态库

1. 静态库

Linux开发工具gcc/g++篇,此篇文章讲过动静态库的基本概念,不了解的可以先看一下这篇文章。

现在我们先来制作一个简单的静态库

mymath.h

#pragma once#include<stdio.h>extern int meyerrno;
int add(int x,int y);
int sub(int x,int y);
int mul(int x,int y);
int div(int x,int y);

mymath.c

#include"mymath.h"int myerrno = 0;
int add(int x,int y)
{return x+y;
}
int sub(int x,int y)
{return x-y;
}
int mul(int x,int y)
{return x*y;
}
int div(int x,int y)
{if(y == 0){myerrno = -1;return -1;}return x/y;
}

我们要是想把我们这个方法提供给不然用,2种方法

  1. 直接将源文件给别人

  2. 将源代码打包成库 -> 库+.h文件

    这里头文件是是必须要给的,这个头文件就相当于我们库的一份说明书

别人要使用我们的方法,其就是将我们的这个.c文件全部编译成.o,然后链接起来形成可执行,现在我们不给源文件,我们自己先编译成.o,然后全部打包给人家,人家就只需要编译自己的.c文件,然后结合我们这个打包文件即可

指令ar -rc lib.o test1.o test2.o将编译好的.o文件打包起来

这时候就已经将库打包好了,要将其发布,我们只需再包装一下即可

将这个打包好的库,使用一下

image-20231115193958352

我们这里包含完整路径,告诉这个头文件在哪儿,但一般我们都是直接包含头文件的名称,不会带上路径

#include"mymath.h"
int main()
{printf("1+1=%d\n",add(1,1));return 0;
}

但是这样的话,编译就不通过了,找不到这个头文件

在gcc中,它会在当前目录和系统里面搜索,这个当前目录指的是和源代码在同一级目录下

image-20231115194238561

我们可以采用gcc main.c -I ./lib/include,告诉gcc去这个路径里面找我们的头文件

这里的路径不需要再跟上我们的mymath.h,因为我们代码中已经包含了

image-20231115194540196

再继续编译,这里没有报找不到头文件的错误,而是报的的是链接错误

这里我们可以测试一下,让他生成目标文件,不进行链接,这里是没有问题的

image-20231115194741297

这是因为找不到我们打包好的静态库,这个原因也是gcc只会去默认的库路径或者是当前目录找库。

所以我们还要带上gcc main.-I ./lib/include/ -L ./lib/mymathlib/,告诉gcc去这个路径找,执行发现,还是错误

image-20231115195408595

报错还是一样,这里是因为我们并没告诉gcc,要链接哪个库,我们后面还要跟上gcc main.c -I ./lib/include/ -L ./lib/mymathlib/ -l mymath

小写-l后面简易紧跟库名称,库的真实名称是去掉前缀,去掉后缀

image-20231115200549943

这些选项带着看着有点冗余,我们之前写的代码是纯C的代码,gcc能在系统中找到这些动静态库。

而我们现在用的库,属于第三方库,所以要带上这些选项,如果想这样做,我们可以将这个库的头文件拷贝到系统的include,将库文件拷贝到系统的lib64目录中,然后指定-l编译,如果不想这样,我们可以建立软链接

Tips:

我们这里写的库中,对除零错误进行了判定

printf("11/0 = %d errno = %d\n",div(11,0),myerrno);

这段代码的myerrno并没有改变,这是因为传参是从右向左传递的,传递myerrno的时候,div函数还未调用,所以myerrno没有改变

当我们链接完毕之后,可采用指令ldd a.out查看所依赖的动态库,这里却并没有mymath.c

image-20231115204125246

gcc默认动态链接,我们这里只提供了静态库,所以gcc只能对这个库进行静态链接。

这里动静态库都链接了,这也说明,如果需要gcc可以同时链接多个动静态库

接下来我们将我们的库文件拷贝到系统路径

image-20231115205743400

然后不带上IL这些选项,编译发现,并没有出现找不到头文件的错误,出现的是链接错误

image-20231115205940209

还是需要链接gcc main.c -lmymath

image-20231115210033429

我们上面的对库进行拷贝,本质上就是对这个三方库进行安装,所谓的下载,就是在系统路径下去掉这个库。

这里第三方库,必须要指明库名称!

当然,这里并不是很建议将自己写的库安装到系统路径,我们也可以采用软连接的方式

sudo ln -s /home/Pyh/linux/study/lib_11_15/test/lib/include /usr/include/myinc
sudo ln -s /home/Pyh/linux/study/lib_11_15/test/lib/mymathlib/libmymath.a /lib64/libmymath.a

image-20231116091609995

image-20231116091822359

这两种方式都可以,但是对于第三方库,我们还是需要安装他们都说明书来进行安装

2. 动态库

不管是形成动态库还是静态库,第一步都是先将源文件生成目标.o文件

image-20231116093631971

采用指令:

gcc -fPIC -c myprint.c
gcc -fPIC -c mylog.c

然后将其打包,ar指令是专门打包静态库的。形成动态库直接采用的是gcc

gcc -shared -o libmymethod.so *.o

生成可执行程序就是将所有的.o文件链接,可是我们的这两个.o里面并没有main函数,都是提供的方法,所以要跟上shared,表示生成共享库(动态库)

image-20231116101050170

这里形成的动态库是有可执行权限的,而静态库是没有的

静态库是提供源代码的,在形成可执行的时候,它的作用就是将需要的内容拷贝过去,并不会加载到内存

对于动态库来说,它要和可执行程序产生关联,当可执行程序运行时,要访问动态库的内容,要跳转到动态库,那么动态库就必须加载到内存当中,所以要有可执行权限

我们可以使用Makefile一次性生成我们的动静态库

dynamic-lib=libmymethod.so
static-lib=libmymath.a.PYONY:all
all:$(dynamic-lib) $(static-lib)$(static-lib):mymath.oar rc $@ $^
mymath.o:mymath.cgcc -c $^$(dynamic-lib):mylog.o myprint.ogcc -shared -o $@ $^
mylog.o:mylog.cgcc -fPIC -c $^
myprint.o:myprint.cgcc -fPIC -c $^.PHONY:clean
clean:rm -rf *.o *.a *.so mylib.PHONY:output
output:mkdir -p mylib/includemkdir -p mylib/libcp *.h mylib/includecp *.a mylib/libcp *so mylib/lib

这个单独使用静态库还是和上面一样,就不做演示了。我们来使用动态库看看,方法也是一样

image-20231116111000244

生成可执行程序没有问题,但是运行的时候报错了。

我们先看一下依赖的动态库ldd a.out

image-20231116113227772

发现这里是not found,我们在file a.out看一下,这个程序是使用共享库的

image-20231116113420002

那么问题就是出现在了这个not found

我们这里是动态库,我们只是告诉了编译器,这个库的路径在哪里。可是动态库是要加载到内存的,系统的加载器并不知道它在哪儿。

这也可以想静态库一样,直接安装到系统路径下,这里就不演示了。

  • 方法1:拷贝到默认库路径

  • 方法2:建立软链接

    sudo ln -s /home/Pyh/linux/study/lib_11_15/test/mylib/lib/ /lib64/libmymethod.so
    

    image-20231116164533102

  • 方法3:将自己库所在的路径,添加到系统环境变量LD_LIBRARY_PATH

    export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/Pyh/linux/study/lib_11_15/test/mylib/lib
    

    image-20231116192936219

    如果重新登录,那么这个配置的环境变量就失效了,要是想长久有效,可以将其添加到系统启动时候的脚本里面~/.bash_profile

    image-20231116213355331

  • 方法4:在目录/etc/ld.so.conf.d里面,加上后缀位.conf的路径文件,然后ldconfing即可(需要root权限)

    image-20231116235237846

    这里不用添加库名字,因为我们这个程序在编译的时候,已经知道了库的名字,所以系统知道链接哪个库

其实在实际情况下,我们用的库都是别人成熟的库,都采用直接安装到系统当中

推荐库ncurses,这个是基于终端的,可提供一些图形化界面的库

3. 动态库的加载

  1. 动态库在程序运行的时候,会被加载到内存当中,而静态库没有;
  2. 创建的动态库被所以的可执行文件动态链接,都要使用,所以动态库也叫共享库。

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

那这个动态库是怎么加载的呢?

动态库也是文件,文件都是存储在磁盘上。如果我们要执行这个程序需要访问这个库的代码,那么这个库就会加载到内存当中。

而进程访问内存是页表映射的方式,那么这个加载到内存的库,就会映射到页表,然后页表再映射到进程地址空间的共享区里面,这时候代码区要访问这个库,就跳转共享区,拿到要的方法,执行完毕之后再返回代码区。所以在建立映射之后,我们执行的任何代码,都是在进程地址空间中执行的。

一个共享库可以给多个进程使用,而一个进程也可以使用多个共享库,这就说明在系统当中,一定会有多个动态库,操作系统要将这些库管理起来,也是六个字先描述,再组织。所以这个共享库有没有加载到内存当中,操作系统是十分清楚的。

这里还有个问题,动态库中,可能会存在全局变量,例如errno这样的,如果我们的进程将其修改了,会不会影响其他的进程呢?

这肯定是不会的,如果这个进程要修改这个全局变量,就需要对这个变量进行写入,写入的过程会发生写时拷贝。这也是进程为什么具有独立性的一个原因!

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

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

相关文章

Apache阿帕奇安装配置

目录 一、下载程序 1. 点击Download 2. 点击Files for Microsoft Windows 3. 点击Apache Lounge 4. 点击httpd-2.4.54-win64-VSI6.zip ​5. 下载压缩包 6.解压到文件夹里 二、配置环境变量 1. 右键我的电脑 - 属性 2. 高级系统设置 3. 点击环境变量 4. 点击系统变…

springboot326校园体育场馆(设施)使用管理网站

交流学习&#xff1a; 更多项目&#xff1a; 全网最全的Java成品项目列表 https://docs.qq.com/doc/DUXdsVlhIdVlsemdX 演示 项目功能演示&#xff1a; ————————————————

【Java】详解多线程通信

&#x1f33a;个人主页&#xff1a;Dawn黎明开始 &#x1f380;系列专栏&#xff1a;Java ⭐每日一句&#xff1a;什么都不做&#xff0c;才会来不及 &#x1f4e2;欢迎大家&#xff1a;关注&#x1f50d;点赞&#x1f44d;评论&#x1f4dd;收藏⭐️ 文章目录 &#x1f510;多…

12v24v60v高校同步降压转换芯片推荐

12V/24V/60V 高校同步降压转换芯片推荐&#xff1a; 对于需要高效、稳定、低噪音的降压转换芯片&#xff0c;推荐使用WD5030E和WD5105。这两款芯片都是采用同步整流技术&#xff0c;具有高效率、低噪音、低功耗等优点&#xff0c;适用于各种电子设备。 WD5030E是一款高效率…

Scrapy----Scrapy简介

文章目录 概述与应用背景架构和组件功能和特点社区生态概述与应用背景 Scrapy,一个高效、灵活、且强大的Web爬取框架,被广泛应用于数据抓取和网页内容的结构化提取。它是用Python编写的,支持多平台运行,适用于数据挖掘、在线零售信息收集、历史数据存档等多种场景。Scrapy…

交换排序详讲:冒泡排序+快速排序(多方法+思路+图解+代码)

文章目录 交换排序一.冒泡排序二.快速排序1.挖坑法2.Hoare法 交换排序 根据序列中两个记录键值的比较结果来对换这两个记录在序列中的位置将键值较大的记录向序列的尾部移动&#xff0c;键值较小的记录向序列的前部移动。 一.冒泡排序 /*** 冒泡排序* 时间复杂度 n^2* 空间复杂…

R语言——taxize(第二部分)

taxize&#xff08;第二部分&#xff09; 3. taxize 文档中译3.10. classification&#xff08;根据类群ID检索分类阶元层级&#xff09;示例1&#xff1a;传递单个ID值示例2&#xff1a;传递多个ID值示例3&#xff1a;传递单个名称示例4&#xff1a;传递多个名称示例5&#xf…

nvm安装详细教程(卸载旧的nodejs,安装nvm、node、npm、cnpm、yarn及环境变量配置)

文章目录 一、完全卸载旧的nodejs1、打开系统的控制面板&#xff0c;点击卸载程序&#xff0c;卸载nodejs&#xff08;1&#xff09;打开系统的控制面板&#xff0c;点击程序下的卸载程序&#xff08;2&#xff09;找到node.js&#xff0c;鼠标右击出现下拉框&#xff0c;点卸载…

yolov5模型代码怎么修改

yaml配置文件 深度乘积因子 宽度乘积因子 所有版本只有这两个参数的不同&#xff0c;s m l x逐渐加宽加深 各种类型层参数对照 backbone里的各层&#xff0c;在这里解析&#xff0c;只需要改.yaml里的各层参数就能控制网络结构 修改网络结构 第一步&#xff1a;把新加的模块…

【7】Spring Boot 3 集成组件:缓存组件 spring cache + spring data redis

目录 【7】Spring Boot 3 集成组件&#xff1a;缓存组件 spring cache spring data redis什么是缓存抽象声明式注解JSR-107对应SpEL上下文数据 引入依赖cache 支持的缓存类型缓存类型配置NONESIMPLEREDIS自定义配置 CAFFEINE Hazelcast...总结 个人主页: 【⭐️个人主页】 需要…

GUI编程--PyQt5--QTreeWidget

文章目录 树型控件展示数据修改节点数据获取所有节点的数据 Qt模组参考 QWidgets QTreeWidget 树型控件展示数据 展示数据的同时&#xff0c;每个节点标注数据类型。 class MyWindow(QWidget):def __init__(self, title):super(MyWindow, self).__init__()self.setWindowTitl…

遵循开源软件安全路线图

毫无疑问&#xff0c;开源软件对于满足联邦任务所需的开发和创新至关重要&#xff0c;因此其安全性至关重要。 OSS&#xff08;运营支持系统&#xff09; 支持联邦政府内的每个关键基础设施部门。 联邦政府认识到这一点&#xff0c;并正在采取措施优先考虑 OSS 安全&#xff…