【Linux】动态库和静态库——动态库和静态库的打包和使用、gcc编译、拷贝到系统默认的路径、建立软连接

文章目录

  • 动态库和静态库
    • 1.静态库和动态库的介绍
    • 2.静态库的打包和使用
      • 2.1生成静态库
      • 2.2使用静态库的三种方式
        • 2.2.1gcc编译
        • 2.2.2拷贝到系统默认的路径
        • 2.2.3建立软连接
    • 3.动态库的打包和使用
      • 3.1生成动态库
      • 3.2使用动态库
      • 3.3解决加载不到动态库的方法

动态库和静态库

1.静态库和动态库的介绍

  静态库和动态库是两种不同的程序库,它们在编译和链接阶段有不同的应用方式和特点。

  静态库(Static Library):

  静态库是在编译时被全部链接到目标程序中,一同生成可执行文件,所以生成的可执行文件较大,但运行时不需要链接其他库。静态库的后缀通常为.a或.lib。在程序发布时,通常只需要提供静态库和可执行文件,而不需要源代码。

  动态库(Dynamic Library):

  动态库在程序运行时才被加载和链接,所以多个程序可以共享相同的动态库代码,从而节省内存。动态库的后缀通常为.so(Linux)或.dll(Windows)。动态库的代码需要满足能够被加载到不同进程的不同地址,因此需要进行特别的编译处理。动态库在程序运行时由操作系统负责加载和链接,因此如果程序需要更新某个模块,只需要更新相应的动态库即可,而不需要重新编译整个程序。

  静态库和动态库的主要区别在于链接时间和使用方式。静态库在编译时链接到目标程序中,而动态库在程序运行时才被加载和链接。此外,静态库和动态库的打包和分发方式也不同,静态库需要和可执行文件一起发布,而动态库只需要提供动态库文件即可。

  静态库(后缀为 .a):程序在编译链接的时候把库的代码链接到可执行文件中。程序运行的时候将不再需要静态库。

  动态库(后缀为 .so):程序在运行的时候才去链接动态库的代码,多个程序共享使用库的代码。

            

2.静态库的打包和使用

  静态库的打包原理基于链接器的工作原理,将各个目标文件中的代码和符号合并到一起,以便在其他项目中进行使用。先将所有目标文件(.c文件)编译为(.o文件),然后把目标文件(.o文件)打包在一起,生成一个或多个静态库文件。 这个过程通常使用ar命令(在Linux和类Unix系统中)或lib命令(在Windows系统中)完成。

  打包过程中,需要将所有的目标文件组织到一个归档文件中,形成静态库。这个归档文件是一个持久的数据库,包含了目标文件的名称、创建时间和修改时间等信息。在链接阶段,链接器会从静态库中提取需要的目标文件,将其链接到最终的可执行文件中。

  

2.1生成静态库

  我们假设使用mymath.c和mymath.h模拟为静态库。下面是制作和打包静态库的过程。

  假设我们的main.c想要编译外部的两个源文件和头文件(mymath.c和mymath.h)。

  mymath.h

#pragma once

#include <stdio.h>

extern int myerrno;

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;
}

  
  main.c

#include "mymath.h"
//#include "myinc/mymath.h"

int main()
{
    extern int myerrno;
    //printf("1+1=%d\n", add(1,1));
    int n=div(10,0);
    //C语言实例化是从右向左,所以myerrno输出的是0
    printf("10/0=%d, errno=%d\n", n, myerrno);

    //gcc默认的链接方式是动态链接
    //没有动态库就默认使用静态库链接

    return 0;
}  

  

  经过下面的make操作,生成.a静态库文件。

在这里插入图片描述

static-lib=libmymath.a//将我们静态库的名称命名为static-lib

$(static-lib):mymath.o//如何使用mymath.o构建static-lib
	ar -rc $@ $^//构建静态库 目标文件 依赖文件
mymath.o:mymath.c//如何使用mymath.c构建mymath.o
	gcc -c $^//编译到.o文件

.PHONY:clean//伪目标
clean:      //清除所有的.o  .a  和static-lib文件
	rm -rf *.o *.a static-lib
	
.PHONY:output//打包文件
output:
	mkdir -p static-lib/include//创建目录include
	mkdir -p static-lib/my-static-lib//创建目录my-static-lib
	cp *.h static-lib/include//拷贝所有.h文件到include
	cp *.a static-lib/my-static-lib//拷贝所有.a文件到my-static-lib

  

  此时的文件为,我们进行make操作:

在这里插入图片描述

  

  我们可以看到生成了mymath.o文件和我们需要的打包好的静态库libmymath.a文件。此时我们就需要使用这个静态库libmymath.a了。

在这里插入图片描述
  

  顺带着打包一下,将.h和.a文件放入一个static-lib文件中。

在这里插入图片描述
在这里插入图片描述

  

2.2使用静态库的三种方式

2.2.1gcc编译

  当前的文件下输入 gcc main.c -I ./头文件的路径 -L ./库文件的路径 -l 链接库的名称 即可生成我们的可编译程序。

在这里插入图片描述
  

  注意上面的代码所含的内容缺一不可:

  缺少头文件和库文件,链接出错。

在这里插入图片描述
  
  缺少库文件,链接出错。

在这里插入图片描述

  
  找不到链接库的名称,链接出错。

在这里插入图片描述

  
  虽然链接出错,但是仍然可以汇编为.o文件。

在这里插入图片描述

  

2.2.2拷贝到系统默认的路径

  拷贝文件到系统路径同样可以实现静态库的使用:

  sudo cp static-lib/include/mymath.h /usr/include/

   sudo cp static-lib/my-static-lib/libmymath.a /lib64/libmymath.a

在这里插入图片描述
  

  gcc 无法直接编译我们的main.c文件还是需要我们告诉编译器其中的静态库的名字才可以,-l mymath

在这里插入图片描述
  

  但是一般不推荐,这样会对我们系统的路径造成污染,删除:

在这里插入图片描述

  

2.2.3建立软连接

  软链接应用广泛,可以快速找到.h和.c文件。

  使用时,main函数的头文件要修改为文件的路径。

//#include "mymath.h"
#include "myinc/mymath.h"//使用软链接时编译

  软链接includesudo ln -s /home/wu1/study_liunx/2024_1_23动静态库测试/static-lib/include /usr/include/myinc

  软链接.a静态库sudo ln -s /home/wu1/study_liunx/2024_1_23动静态库测试/static-lib/my-static-lib/libmymath.a /lib64/libmymath.a

  解除链接:sudo unlink/usr/include/myinc sudo unlink/lib64/libmymath.a

在这里插入图片描述

            

3.动态库的打包和使用

  动态库的打包原理是将多个相对独立的部分按照模块化的方式拆分成不同的文件,并在程序运行时才将这些模块链接在一起形成一个完整的程序。与静态库不同,动态库不会将所有代码和数据都包含在最终的可执行文件中,而是在程序运行时由操作系统动态加载到内存中。

  打包动态库时,需要将各个目标文件(.o文件)编译为动态库文件(.so文件),以便在程序运行时被加载和链接。 这个过程通常使用gcc命令,并指定-fPIC-shared选项,以便生成位置无关代码和共享库。

  

3.1生成动态库

  和上面生成的.a类似,动态库是后缀为.so的文件,我们下面使用mylog.h mylog.c myprint.h myprint.c进行动态库的打包实现。

  main.c

#include "mylog.h"
#include "myprint.h"

int main()
{
    Print();
    Log("这是一个动态库打包的测试");

    return 0;
}  

  
  mylog.h

#pragma once

#include <stdio.h>

void Log(const char*);

  
  mylog.c

#include "mylog.h"

void Log(const char*info)
{
    printf("Warning: %s\n", info);
}

  
  myprint.h

#pragma once

#include <stdio.h>

void Print();

  
  myprint.c

#include "myprint.h"

void Print()
{
    printf("hello new world!\n");
    printf("hello new world!\n");
    printf("hello new world!\n");
    printf("hello new world!\n");
}

  

  经过下面的make操作,生成.so静态库文件。
在这里插入图片描述

dy-lib=libmymethod.so//将我们动态库的名称命名为dy-lib

.PHONY:all//伪目标为dy-lib文件
all: $(dy-lib)

$(dy-lib):mylog.o myprint.o//将mylog.o和myprint.o文件打包为动态库文件
	gcc -shared -o $@ $^//形成共享库(可执行程序加载内存)

mylog.o:mylog.c//将.c文件编译为.o文件
	gcc -fPIC -c $^//-fPIC产生与位置无关码
myprint.o:myprint.c
	gcc -fPIC -c $^

.PHONY:clean//伪目标删除操作
clean:
	rm -rf *.o *.so dy-lib

.PHONY:output//打包动态库
output:
	mkdir -p dy-lib/include
	mkdir -p dy-lib/my-dy-lib
	cp *.h dy-lib/include
	cp *.so dy-lib/my-dy-lib

  

  此时的文件为,我们进行make操作:

在这里插入图片描述
  

  我们将我们的头文件和.so文件打包为了dy-lib文件。

在这里插入图片描述

在这里插入图片描述
  

3.2使用动态库

  和上面使用静态库一样,我们链接头文件和库文件,而且找到链接库的名称即可。

  gcc main.c -I ./dy-lib/include/ -L ./dy-lib/my-dy-lib -l mymethod

  

  但是在链接的时候,会报错。因为动态库在哪里也要告诉系统——加载器,加载同样也需要过程。

在这里插入图片描述
  

  进行动态库的软链接,ldd成功找到链接。

在这里插入图片描述

  

  运行成功。

在这里插入图片描述

  

3.3解决加载不到动态库的方法

  1.拷贝到系统默认的库路径 /lib64 /usr/lib64/

  2.在系统默认的库路径 /ib64 /usr/lib64/下建立软连接

  3.将自己的库所在的路径,添加到系统的环境变量LD LIBRARY PATH中

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

  实际情况,我们用的库都是别人的成熟的库,都采用直接安装到系统的方式。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

1月全志芯片开源项目分享合辑

1、柚子爱AI相机&#xff08;YuzuAI-YuzuMaix-AIoT-V831&#xff09; 本项目于去年4月首次发布&#xff0c;是基于V831的AI相机开源项目&#xff0c;经过几个版本的迭代&#xff0c;最终运用了叠层的设计来实现AI摄像头掌控板的奇葩组合。 开发板主控是全志V831&#xff0c;采…

基于SpringBoot+Redis的前后端分离外卖项目-苍穹外卖微信小程序端(十五)

用户端历史订单模块 1. 查询历史订单1.1 需求分析和设计1.2 代码实现1.2.1 user/OrderController1.2.2 OrderService1.2.3 OrderServiceImpl1.2.4 OrderMapper1.2.5 OrderMapper.xml1.2.6 OrderDetailMapper 2. 查询订单详情2.1 需求分析和设计2.2 代码实现2.2.1 user/OrderCon…

qt初入门7:进度条,定时器,时间控件练习

参考课本demo&#xff0c;空闲时间练习一下进度条&#xff0c;定时器&#xff0c;日期相关控件和使用。 1&#xff1a;demo运行结果 2&#xff1a;进度条控件梳理 进度条显示控件实际上是QProgressBar, 显示的进度可以通过代码控制&#xff0c;也可以通过其他控件上获取到的值…

【算法】BFS算法解决多源最短路问题(C++)

文章目录 前言那么什么是单源最短路 / 多源最短路呢&#xff1f;如何解决此类题&#xff1f;解法一解法二对于解法二&#xff0c;如何编写代码&#xff1f; 算法题542.01矩阵1020.飞地的数量1765.地图中的最高点1162.地图分析 前言 此前我们对 单源最短路 问题进行的讲解&…

Linux中并发程序设计(进程的创建和回收、exec函数使用、守护进程创建和使用、GDB的父、子进程代码的调试、线程的创建和参数传递)

进程的创建和回收 进程概念 概念 程序 存放在磁盘上的指令和数据的有序集合&#xff08;文件&#xff09; 静态的 进程 执行一个程序所分配的资源的总称 动态的进程和程序比较 注&#xff1a;进程是存在RAM中&#xff0c;程序是存放在ROM(flash)中的进程内容 BSS段&#xff…

一篇文章搞懂Python的文件读写

目录 一、打开文件 二、读取文件内容 三、写入文件内容 四、追加内容到文件末尾 五、文件路径 六、错误处理 七、关闭文件 八、文件读写的高级功能 总结 在Python中&#xff0c;文件读写是一项常见的操作。通过文件读写&#xff0c;我们可以保存数据&#xff0c;加载数…

算法学习记录:动态规划

前言&#xff1a; 算法学习记录不是算法介绍&#xff0c;本文记录的是从零开始的学习过程&#xff08;见到的例题&#xff0c;代码的理解……&#xff09;&#xff0c;所有内容按学习顺序更新&#xff0c;而且不保证正确&#xff0c;如有错误&#xff0c;请帮助指出。 学习工具…

Find My资讯|苹果Apple Pencil 3将支持Find My定位查找功能

国外科技媒体 9to5Mac 深挖 iOS 17.4 Beta 1 更新代码&#xff0c;发现了 Apple Pencil 3 的踪迹&#xff0c;并显示该手写笔支持 Find My 功能。这就意味着可以和 AirPods 和 AirTags 一样&#xff0c;用户可以通过 Find My 网络&#xff0c;寻找丢失的 Apple Pencil 手写笔。…

Kotlin for loop: in、 until、 step、 downTo

Kotlin for loop: in、 until、 step、 downTo fun loop1() {for (i in 0..5) {print("$i ")}println("\n1-end\n") }fun loop2() {for (i in 0 until 5) {print("$i ")}println("\n2-end\n") }fun loop3() {for (i in 0 until (5)) {…

探索Viper-适用于GoLang的完整配置解决方案

前言 对于现代应用程序&#xff0c;尤其大中型的项目来说&#xff0c;在程序启动和运行时&#xff0c;往往需要传入许多参数来控制程序的行为&#xff0c;我们可以通过命令行参数&#xff0c;环境变量&#xff0c;配置文件等方式来将参数传递给程序。而Viper库为Golang语言开发…

LabVIEW继电器触点接触电阻自动测试

继电器作为工业中的重要组件&#xff0c;其性能直接影响着整个生产线的可靠性和安全性。触点接触电阻是衡量继电器性能的重要参数&#xff0c;传统的测试方法效率低下且成本高昂。为了解决这些问题&#xff0c;采用LabVIEW软件&#xff0c;结合专业的硬件平台&#xff0c;实现了…

备战蓝桥杯----数据结构及STL应用(基础2)

上次我们讲了vector的大致内容&#xff0c;接下来让我们讲一下栈&#xff0c;队列吧&#xff01; 什么是栈呢&#xff1f; 很简单&#xff0c;我们用的羽毛球桶就是&#xff0c;我们取的球&#xff0c;是最后放的&#xff0c;栈是一种先进后出的数据结构。 方法函数 s.push(…

怎样做好Code Review

Code Review方案 定义 Code Review代码评审是指在软件开发过程中&#xff0c;通过对源代码进行系统性检查的过程。通常的目的是查找各种缺陷&#xff0c;包括代码缺陷、功能实现问题、编码合理性、性能优化等&#xff1b;保证软件总体质量和提高开发者自身水平 code review …

SRM是什么意思?SRM供应商管理系统哪个好?

阅读本文&#xff0c;您将了解&#xff1a;一、SRM是什么意思&#xff1b;二、SRM的应用价值&#xff1b;三、SRM供应商管理系统哪个好&#xff1f;推荐在零代码平台自主开发。 在当今快速变化的商业环境中&#xff0c;企业之间的协作和效率至关重要。供应链管理作为企业运营的…

HarmonyOS鸿蒙学习笔记(24)AppScope的icon和UIAblity的icon的区别

在HarmonyOS中有两个配置文件app.json5配置文件和module.json5配置文件&#xff0c;两个配置文件都有icon和label标签&#xff1a; 二者区别如下&#xff1a; app.json5的icon app.json5里面的icon和label:对应的是应用图标和应用标签&#xff0c;应用图标和标签是在设置应用…

HarmonyOS --@state状态装饰器

在声明式UI中&#xff0c;是以状态驱动视图更新。 状态&#xff08;state&#xff09;&#xff1a;指驱动视图更新的数据&#xff08;被装饰器标记的变量&#xff09;。 试图&#xff08;view&#xff09;&#xff1a;基于UI描述渲染得到用户界面 State装饰器标记的变量必须初…

C# Socket 允许控制台应用通过防火墙

需求&#xff1a; 在代码中将exe添加到防火墙规则中&#xff0c;允许Socket通过 添加库引用 效果&#xff1a; 一键三联 若可用记得点赞评论收藏哦&#xff0c;你的支持就是写作的动力。 源地址: https://gist.github.com/cstrahan/513804 调用代码: private static void …

雅特力AT32 Workbench图形化代码生成工具,简化嵌入式开发利器

嵌入式系统应用市场广泛&#xff0c;早已遍及日常生活&#xff0c;随着产品需求复杂度的提升&#xff0c;32位MCU开发难度也随之增加&#xff0c;如何降低开发成本&#xff0c;缩短开发周期&#xff0c;是所有嵌入式开发人员的共同课题。 面对市场竞争日益加剧的情形&#xff…

【Web前端实操17】导航栏效果——滑动门

滑动门 定义: 类似于这种: 滑到导航栏的某一项就会出现相应的画面,里面有对应的画面出现。 箭头图标操作和引用: 像一些图标,如果需要的话,可以找字体图标,比如阿里巴巴矢量图标库:iconfont-阿里巴巴矢量图标库 选择一个——>添加至购物车——>下载代码 因…

实战 | OpenCV+OCR实现弧形文字识别实例(详细步骤 + 源码)

导 读 本文主要介绍基于OpenCV+OCR实现弧形文字识别实例,并给详细步骤和代码。源码在文末。 背景介绍 测试图如下,目标是正确识别图中的字符。图片来源: https://www.51halcon.com/forum.php?mod=viewthread&tid=6712 同样,论坛中已经给出了Halcon实现代码,…