动静态库-Linux 学习
在软件开发中,程序库是一组预先编写好的程序代码,它们存储了常用的函数、变量和数据结构等。这些库可以帮助开发者节省大量的时间和精力,避免重复编写相同的代码。当我们在 Linux 系统中开发程序时,经常会用到两种类型的程序库:静态库和动态库。
- 静态库 (
.a):静态库是指在程序编译时,将库文件中的代码直接复制到可执行文件中。这意味着当程序运行时,库文件中的代码已经包含在可执行文件中,不需要在运行时进行加载。 - 动态库 (
.so):动态库是指在程序运行时才加载的库文件。程序本身不包含库文件的代码,而是通过链接到动态库来调用其中的函数和数据结构等。动态库通常以共享库的形式存在,多个程序可以共享一个动态库。
从库的制作者角度
静态库
我们可以自己尝试写一个简单的计算程序来实现简单的加减乘除功能。
我们可以先创建以下文件:
加法:Add.c , Add.h
减法:Sub.c , Sub.h
乘法:Mul.c , Mul.h
除法:Div.c , Divh

如果我们要上面的文件形成一个库,我们是不能在上面的这些文件中写main函数的,因为库中是没有main 函数的,我们也不能把main 函数打入库中。
// Add.h
#pragma once // 防止头文件被重复包含
#include <stdio.h>
int Add(int,int);
----------------------------
// Add.c
#include "Add.h"
int Add(int x, int y)
{return x + y;
}
// Sub.h
#pragma once
#include <stdio.h>
int Sub(int,int);
----------------------------
// Sub.c#include "Sub.h"
int Sub(int x, int y)
{return x - y;
}
//Mul.h
#pragma once
#include <stdio.h>
int Mul(int,int);
----------------------------
// Mul.c#include "Mul.h"
int Mul(int x, int y)
{return x * y;
}
//Div.h
#pragma once
#include <stdio.h>
int Div(int,int,int*);
------------------------------
// Div.c#include "Div.h"
int Div(int x, int y, int *code)
{*code = 0;if (y == 0){*code = -1;return -1; }return x / y;
}
我们现在要将上面的一堆的头文件 和 源文件 形成静态库
为 我们的方法进行测试,我们可以建立一个叫TestMain.c 的文件。
// TestMain.c
#include "Add.h"
#include "Sub.h"
#include "Mul.h"
#include "Div.h"int x = 20;int y = 10;printf("%d + %d = %d\n",x, y, Add(x,y));printf("%d - %d = %d\n",x, y, Sub(x,y));printf("%d * %d = %d\n",x, y, Mul(x,y));
我们可以运行一下上面的测试程序TestMain.c,不过我们要先进行编译:
// Linux 系统 gcc -o test Add.c Sub.c Mul.c Div.c TestMain.c// 我们在编译的时候没有加上头文件,因为当前的头文件和源文件是在同一个路径下的,所以我们不用加上头文件。
上面我们是把所有的源文件都进行了编译的,其实如果我们要形成一个可执行程序,我们不太建议将所有的源文件都直接进行编译,因为我们一旦这样做,每一个源文件都要进行预处理,编译,汇编,链接。所以一般在面对这样的多文件项目时我们一般都建议把这样的源文件编译成.o 文件。编译成.o 文件之后,然后我们在把所有的.o 文件进行对应的链接 形成一个可执行。
.c–(预处理,编译,汇编)—>.o----(链接)----> 可执行
// 介绍一下`-c` 的选项。
gcc -c test.c // 在 test.c形成可执行前(即:链接前) 终止,最终会生成一个`.o` 的文件,默认是同名的`test.o`
所以如果我们把上面文件中的TestMain.o 去掉,为什么要去掉呢?因为如果我们要做一个静态库,库中是不能有main 函数的,main 函数只能是用这个库的用户写的。这样一来就只剩下Add.0,Sub.o,Mul.o,Div.o 了,我们只需要对Add.0,Sub.o,Mul.o,Div.o 这些.o 文件进行打包,以后如果某个用户想要用到这些加减乘除的方法,只需要将自己写的main 函数编译成xxx.o 文件,然后再和打包好的文件中的.o 文件进行链接就好了。
生成静态库
那我们如何对我们的.o文件进行打包呢?
ar 命令
ar命令是一个把所有的.o 文件打包形成我们对应的库文件的过程。
ar -rc libmymath.a add.o sub.o
ar是gnu归档工具,选项:rc表示(replace and create),他的意思是:把所有的.o文件打包生成一个.a 文件,如果存在就替换(replace)如果不存在就创建(create)。
库的名字要以lib 开头,以.a 结尾,所以上面我们的库是mymath ,但是加上前缀和后缀之后就是:libmymath.a 了。
静态库的原理就是将库中的源代码直接翻译成
.o目标二进制文件,然后打包
动态库
一、什么是动态库?
动态库,也叫共享库(Shared Library),是一种在程序运行时才被加载到内存中的代码库。它的文件名通常以 libxxx.so 的形式命名(xxx 是库的名字,so 表示 shared object)。
与动态库相对的是静态库(Static Library),静态库在程序编译时会被直接嵌入到可执行文件中,文件名通常以 libxxx.a 的形式命名。
二、生成动态库
1. 编译选项
生成动态库需要使用 gcc 编译器,并指定以下选项:
-fPIC:生成位置无关代码(Position Independent Code, PIC)。这种代码可以在内存中的任何位置运行,是动态库的必要条件。-shared:表示生成共享库格式。
2. 示例代码
假设我们有以下文件:
add.c:实现加法函数。sub.c:实现减法函数。
文件内容如下:
// add.c
#include <stdio.h>int add(int a, int b) {return a + b;
}
// sub.c
#include <stdio.h>int sub(int a, int b) {return a - b;
}
3. 生成动态库的步骤
# 编译源文件为位置无关代码
gcc -fPIC -c add.c sub.c# 生成动态库
gcc -shared -o libmymath.so add.o sub.o
执行完上述命令后,会生成一个名为libmymath.so 的动态库文件。
从库的使用者角度
静态库的使用
上面我们自己写的库libmymath.a 是第三方库,gcc 默认不认识 。所以我们在使用gcc D的时候还要加上一个-l 选项来链接到我们的库,还记得上面我们所说的吗,我们的库是以lib为前缀的,以.a 为后缀的,但是在用-l 链接的时候,我们要去掉前缀和后缀,直接用mymath 像这样:-lmymath ,我们还要用-L 来制定库的路径,我们这样来表示-L. ,用. 来表示当前路径,如果你的静态库在当前路径下的话,所以连在一起就是:
[root@localhost linux]# gcc TestMain.c -L. -lmymath
- TestMain.c 是用户写的程序。
- -L 指定库路径
- -l 指定库名
-I: 指定头文件的路径
如果我们要把自己写的库交给其他用户使用的话,那我们就要把.h 和.a 交给他,
gcc 默认是动态链接的,
动态库的使用
使用动态库
1. 编译选项
在使用动态库时,需要指定以下选项:
-L:指定动态库所在的路径(如果动态库不在系统默认路径中)。-l:指定要链接的动态库的名字(去掉lib前缀和.so后缀)。
2. 示例代码
假设我们有一个主程序 main.c,它调用了动态库中的函数:
// main.c
#include <stdio.h>int add(int a, int b);
int sub(int a, int b);int main()
{printf("add(10, 20) = %d\n", add(10, 20));printf("sub(100, 20) = %d\n", sub(100, 20));return 0;
}
3. 编译主程序
# 编译主程序
gcc main.c -o main -L. -lmymath
-L.表示动态库在当前目录下。-lmymath表示链接libmymath.so动态库。
运行动态库
方法一:将 .so 文件拷贝到系统共享库路径
系统默认的动态库路径通常是 /usr/lib 或 /usr/local/lib或是:/lib64 。可以将动态库文件拷贝到这些路径下:
# 拷贝动态库到 /usr/local/lib
cp libmymath.so /usr/local/lib/
方法二:通过使用软连接 查找动态库
ln -s /*动态库的路径*/ libmymath.so
/*动态库的路径*/ : 这里填动态库的路径
libmymath.so : 这个是软连接的名字。
这样,我们通过在当前目录下建立软连接的方式找到我们对应的库。
gcc 可以在当前目录下找到动态库。所以我们可以在当前目录下建立对应库的软连接,这样来让 gcc 找到,我们也可以在系统的指定目录下建立对应的软连接。
方法三:设置 LD_LIBRARY_PATH 环境变量(LD 的意思就是 load -加载,所以这句话就是 : 加载库路径环境变量)
如果不想将动态库文件拷贝到系统路径,可以通过设置 LD_LIBRARY_PATH 环境变量来指定动态库的路径:
# 设置 LD_LIBRARY_PATH
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/*这里填你写的动态库的路径*/
但是这里导入的环境变量是内存级的,所以你重启电脑之后就没了。
方法四:直接更改系统配置文件:使用 ldconfig 配置动态库路径
如果动态库文件存放在非默认路径下,可以通过修改 /etc/ld.so.conf 文件或在 /etc/ld.so.conf.d/ 下添加配置文件来指定动态库路径。
例如:
# 创建配置文件
echo "/path/to/your/library" > /etc/ld.so.conf.d/mylib.conf# 更新动态库缓存
ldconfig
同一组库,提供动静两种库,gcc默认使用动态库。
五、使用外部库
Linux 系统中有很多外部库,例如:
ncurses:用于处理屏幕显示和用户输入的库。glib:用于通用工具函数的库。openssl:用于加密和安全通信的库。
使用外部库的步骤与使用自己编写的动态库类似,需要:
- 安装外部库(通常通过包管理工具,如
apt或yum)。
比如安装ncurses库
sudo yum install -y ncurses-devel - 在编译时指定库的路径和名称。
例如,使用 ncurses 库:
# 编译
gcc main.c -o main -lncurses# 运行
./main
相关文章:
动静态库-Linux 学习
在软件开发中,程序库是一组预先编写好的程序代码,它们存储了常用的函数、变量和数据结构等。这些库可以帮助开发者节省大量的时间和精力,避免重复编写相同的代码。当我们在 Linux 系统中开发程序时,经常会用到两种类型的程序库&am…...
【Hudi-SQL DDL创建表语法】
CREATE TABLE 命令功能 CREATE TABLE命令通过指定带有表属性的字段列表来创建Hudi Table。 命令格式 CREATE TABLE [ IF NOT EXISTS] [database_name.]table_name[ (columnTypeList)]USING hudi[ COMMENT table_comment ][ LOCATION location_path ][ OPTIONS (options_lis…...
HTML label 标签使用
点击 <label> 标签通常会使与之关联的表单控件获得焦点或被激活。 通过正确使用 <label> 标签,可以使表单更加友好和易于使用,同时提高整体的可访问性。 基本用法 <label> 标签通过 for 属性与 id 为 username 的 <input> 元素…...
bge-large-zh-v1.5 与Pro/BAAI/bge-m3 区别
ge-large-zh-v1.5 和 Pro/BAAI/bge-m3 是两种不同的模型,主要区别在于架构、性能和应用场景。以下是它们的对比: 1. 模型架构 bge-large-zh-v1.5: 基于Transformer架构,专注于中文文本的嵌入表示。 参数量较大,适合处…...
JVM常用概念之对象初始化的成本
在JVM常用概念之新对象实例化博客中我讲到了对象的实例化,主要包含分配(TLAB)、系统初始化、用户初始化,而我在JVM常用概念之线程本地分配缓冲区(ThreadLocal Allocation Buffer,TLAB)博客中也讲…...
[AI机器人] Web-AI-Robot机器人前瞻版--比奇堡海之霸凯伦
文章目录 简述开源Web-AI-Robot 项目-比奇堡-海之霸-凯伦 技术架构效果预览 简述 本项目配合前端项目bikini_bottom_karen_ui运行,来源于柒杉工作室(截止2025.2,目前我自己)。 打造一个只需要在浏览器上运行的AI智能机器人&#…...
嵌入式学习-EXTI外部中断
STM32 是一种基于 ARM Cortex-M 内核的微控制器系列,广泛应用于嵌入式系统开发。中断(Interrupt)是 STM32 中一个非常重要的功能,它允许微控制器在执行主程序的同时,响应外部事件或内部事件的请求,从而实现…...
CSS—元素水平居中:2分钟掌握常用的水平居中
个人博客:haichenyi.com。感谢关注 1. 目录 1–目录2–行内元素水平居中3–块级元素水平居中 2. 行内元素水平居中 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" …...
PyTorch 中结合迁移学习和强化学习的完整实现方案
结合迁移学习(Transfer Learning)和强化学习(Reinforcement Learning, RL)是解决复杂任务的有效方法。迁移学习可以利用预训练模型的知识加速训练,而强化学习则通过与环境的交互优化策略。以下是如何在 PyTorch 中结合…...
大语言模型学习--本地部署DeepSeek
本地部署一个DeepSeek大语言模型 研究学习一下。 本地快速部署大模型的一个工具 先根据操作系统版本下载Ollama客户端 1.Ollama安装 ollama是一个开源的大型语言模型(LLM)本地化部署与管理工具,旨在简化在本地计算机上运行和管理大语言模型…...
Linux:vim快捷键
Linux打开vim默认第一个模式是:命令模式! 命令模式快捷键操作: gg:光标快速定位到最开始 shift g G:光标快速定位到最结尾 n shift g n G:光标快速定位到第n行 shift 6 ^:当前行开始 …...
Unity 对象池技术
介绍 是什么? 在开始时初始化若干对象,将它们存到对象池中。需要使用的时候从对象池中取出,使用完后重新放回对象池中。 优点 可以避免频繁创建和销毁对象带来性能消耗。 适用场景 如果需要对某种对象进行频繁创建和销毁时,例…...
算法1-4 凌乱的yyy / 线段覆盖
题目描述 现在各大 oj 上有 n 个比赛,每个比赛的开始、结束的时间点是知道的。 yyy 认为,参加越多的比赛,noip 就能考的越好(假的)。 所以,他想知道他最多能参加几个比赛。 由于 yyy 是蒟蒻,…...
【计网】数据链路层
数据链路层 3.1 数据链路层概述3.2 封装成帧3.3 差错检测3.4 可靠传输3.4.1 可靠传输的概念3.4.2 可靠传输的实现机制 - 停止等待协议3.4.3 可靠传输的实现机制 -回退N帧协议3.4.4 可靠传输的实现机制 -选择重传协议 3.5 点对点协议3.5.1 帧格式3.5.2 透明传输 3.6 媒体接入控制…...
javaweb自用笔记:Vue
Vue 什么是vue vue案例 1、引入vue.js文件 2、定义vue对象 3、定义vue接管的区域el 4、定义数据模型data 5、定义视图div 6、通过标签v-model来绑定数据模型 7、{{message}}直接将数据模型message展示出来 8、由于vue的双向数据绑定,当视图层标签input里的…...
CSS Overflow 属性详解
CSS Overflow 属性详解 在网页设计和开发中,CSS Overflow 属性是一个非常重要的特性,它决定了当内容超出其容器大小时应该如何处理。本文将详细介绍 CSS Overflow 属性的相关知识,包括其语法、作用、常用属性值以及一些实际应用场景。 1. CSS Overflow 属性概述 CSS Over…...
沃丰科技结合DeepSeek大模型技术落地与应用前后效果对比
技术突破:DeepSeek算法创新,显著降低了显存占用和推理成本。仅需少量标注数据即可提升推理能力。这种突破减少了对海量数据的依赖,削弱了数据垄断企业的优势! 商业模式颠覆:DeepSeek选择完全开源模式,迫使…...
突破光学成像局限:全视野光学血管造影技术新进展
全视野光学血管造影(FFOA)作为一种实时、无创的成像技术,能够提取生物血液微循环信息,为深入探究生物组织的功能和病理变化提供关键数据。然而,传统FFOA成像方法受到光学镜头景深(DOF)的限制&am…...
2.反向传播机制简述——大模型开发深度学习理论基础
在深度学习开发中,反向传播机制是训练神经网络不可或缺的一部分。它让模型能够通过不断调整权重,从而将预测误差最小化。本文将从实际开发角度出发,简要介绍反向传播机制的核心概念、基本流程、在现代网络中的扩展,以及如何利用自…...
机器学习校招面经二
快手 机器学习算法 一、AUC(Area Under the ROC Curve)怎么计算?AUC接近1可能的原因是什么? 见【搜广推校招面经四】 AUC 是评估分类模型性能的重要指标,用于衡量模型在不同阈值下区分正负样本的能力。它是 ROC 曲线…...
C语言泛型编程与类型安全 - C11的高级特性
引言 C语言通常被认为不支持泛型编程,但实际上通过巧妙的设计模式和C11标准的新特性,我们可以在C语言中实现类型安全的泛型代码。 本文将深入讲解如何使用void指针、宏技巧和C11的_Generic关键字实现泛型编程,让你的代码更加灵活和可复用。 一、void指针泛型基础 1.1 vo…...
堆叠集成方法
原文:towardsdatascience.com/the-stacking-ensemble-method-984f5134463a 发现堆叠在机器学习中的力量——一种将多个模型组合成一个单一强大预测器的技术。本文从基础知识到高级技术探讨了堆叠,揭示了它是如何结合不同模型的优势以提高准确性的。无论你…...
别再只跑测试了!用KAIR库从零训练你自己的SwinIR超分模型(附DIV2K/Flickr2K数据集处理避坑指南)
从测试到训练:SwinIR超分模型实战进阶指南 当你第一次用SwinIR的预训练模型将模糊照片变得清晰时,那种惊艳感可能让你跃跃欲试想训练自己的模型。但面对几十GB的数据集和复杂的训练配置,很多开发者停在了"只跑测试"的阶段。本文将带…...
异构多核处理器如何实现安卓、Linux与RTOS的原生融合?
1. 项目概述:一颗“三栖”处理器的诞生最近在嵌入式圈子和一些硬件开发者社区里,一个话题的热度悄然攀升:一颗号称能同时原生运行安卓、Linux和RTOS的国产CPU。这听起来有点像是“瑞士军刀”式的处理器,试图用一个硬件平台覆盖从消…...
CNAS实验室一份完整的质量手册需要包含哪些要素?一文教会质量手册编写
编写质量管理体系文件是CNAS实验室认证工作中非常重要的一个环节,实验室质量管理体系文件按照惯例,一般会分为四个层级,质量手册、程序文件、作业指导书和记录文件。实验室质量手册是实验室依据相关标准制定的纲领性文件,系统规定…...
SuperRDP完整指南:一键解锁Windows远程桌面多用户并发连接限制
SuperRDP完整指南:一键解锁Windows远程桌面多用户并发连接限制 【免费下载链接】SuperRDP Super RDPWrap 项目地址: https://gitcode.com/gh_mirrors/su/SuperRDP SuperRDP是基于RDPWrap技术的智能工具,专为突破Windows系统远程桌面功能限制而设计…...
MSP430单片机低功耗设计实战:从架构到代码的灵活性解析
1. 项目概述:为什么是MSP430?如果你在嵌入式领域摸爬滚打了一段时间,尤其是在对功耗极其敏感的应用场景里,比如智能穿戴、便携医疗设备、无线传感器网络或者那些需要电池供电数年的工业传感器,那么“MSP430”这个名字对…...
高级磁盘空间管理:WinDirStat深度配置与自动化清理指南
高级磁盘空间管理:WinDirStat深度配置与自动化清理指南 【免费下载链接】windirstat WinDirStat is a disk usage statistics viewer and cleanup tool for Microsoft Windows 项目地址: https://gitcode.com/gh_mirrors/wi/windirstat 在当今数据爆炸的时代…...
Cadence Allegro 16.6 环境设置保姆级教程:从绘图参数到自动保存,新手避坑指南
Cadence Allegro 16.6 环境设置实战指南:从零配置到高效设计 第一次打开Cadence Allegro 16.6时,满屏的菜单选项和参数设置可能会让新手感到无所适从。作为一款专业的PCB设计工具,Allegro提供了高度可定制的工作环境,但这也意味着…...
Perplexity本地化查询实战:手把手教你用Ollama+Llama3构建离线知识库(含性能压测数据)
更多请点击: https://intelliparadigm.com 第一章:Perplexity本地服务查询 Perplexity 本地服务查询是指在不依赖云端 API 的前提下,通过本地部署的模型与推理服务(如 Ollama、LM Studio 或 Text Generation WebUI)完…...
