Linux笔记---动静态库(使用篇)

news2025/3/30 19:59:45

目录

1. 库的概念

2. 静态库(Static Libraries)

2.1 静态库的制作

2.2 静态库的使用

2.2.1 显式指定库文件及头文件路径

2.2.2 将库文件安装到系统目录

2.2.3 将头文件安装到系统目录

3. 动态库

3.1 动态库的制作

3.2 动态库的使用

3.2.1 显式指定库文件路径

2.2.2 将路径加载到环境变量中 

2.2.3 配置文件

4. 总结与补充


1. 库的概念

库(Library) 是一组预先编译好的代码(函数、类、数据等)的集合,可以被多个程序共享和重复使用。库的核心目的是代码复用,避免开发者重复编写相同的功能(如文件操作、数学计算等)。

本质上来说库是一种可执行代码的二进制形式,可以被操作系统载如内存执行。

按照代码复用的形式,库可以分为两种:

  • 静态库:.a [Linux].lib [Windows]
  • 动态库:.so [Linux]、.dll [Windows]

库是在链接这一步被使用的,实际上就是一堆 .o 文件的集合,我们可以特定的工具来将这些 .o 文件进行打包,进而形成库。 

为举例方便,这里给出我们自己实现的简单的C语言库---myc:

// mystdio.h

#pragma once
#include <stdio.h>
#define MAX 1024
#define NONE_FLUSH (1<<0)
#define LINE_FLUSH (1<<1)
#define FULL_FLUSH (1<<2)

typedef struct IO_FILE
{
    int fileno;
    int flag;
    char outbuffer[MAX];
    int bufferlen;
    int flush_method;
}MyFile;


MyFile *MyFopen(const char *path, const char *mode);
void MyFclose(MyFile *);
int MyFwrite(MyFile *, void *str, int len);
void MyFFlush(MyFile *);



// mystdio.c

#include "mystdio.h"
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>

static MyFile *BuyFile(int fd, int flag)
{
    MyFile *f = (MyFile*)malloc(sizeof(MyFile));
    if(f == NULL) return NULL;
    f->bufferlen = 0;
    f->fileno = fd;
    f->flag = flag;
    f->flush_method = LINE_FLUSH;
    memset(f->outbuffer, 0, sizeof(f->outbuffer));
    return f;
}

MyFile *MyFopen(const char *path, const char *mode)
{
    int fd = -1;
    int flag = 0;
    if(strcmp(mode, "w") == 0)
    {
        flag = O_CREAT | O_WRONLY | O_TRUNC;
        fd = open(path, flag, 0666);
    }
    else if(strcmp(mode, "a") == 0)
    {
        flag = O_CREAT | O_WRONLY | O_APPEND;
        fd = open(path, flag, 0666);
    }
    else if(strcmp(mode, "r") == 0)
    {
        flag = O_RDWR;
        fd = open(path, flag);
    }
    else
    {
        //TODO
    }
    if(fd < 0) return NULL;
    return BuyFile(fd, flag);
}
void MyFclose(MyFile *file)
{
    if(file->fileno < 0) return;
    MyFFlush(file);
    close(file->fileno);
    free(file);
}
int MyFwrite(MyFile *file, void *str, int len)
{
    // 1. 拷贝
    memcpy(file->outbuffer+file->bufferlen, str, len);
    file->bufferlen += len;
    // 2. 尝试判断是否满足刷新条件!
    if((file->flush_method & LINE_FLUSH) && file->outbuffer[file->bufferlen-1] == '\n')
    {
        MyFFlush(file);
    }
    return 0;
}
void MyFFlush(MyFile *file)
{
    if(file->bufferlen <= 0) return;
    // 把数据从用户拷贝到内核文件缓冲区中
    int n = write(file->fileno, file->outbuffer, file->bufferlen);
    (void)n;
    fsync(file->fileno);
    file->bufferlen = 0;
}



// mystring.h

#pragma once
int my_strlen(const char *s);



// mystring.c

#include "mystring.h"

int my_strlen(const char *s)
{
    const char *start = s;
    while(*s)
    {
        s++;
    }
    return s - start;
}

接下来,我们会介绍如何将上述的原文件打包成动静态库并使用。 

2. 静态库(Static Libraries)

  • 文件扩展名:.a(Archive)

  • 特点:
    • 在编译时,库的代码会被直接复制到最终的可执行文件中。

    • 生成的可执行文件独立,不依赖运行时环境中的库文件。

    • 缺点:文件体积较大,且更新库时需要重新编译程序。

  • 创建工具:ar(归档工具)+ ranlib(生成索引)。

  • 使用场景:适合对程序独立性要求高的场景。

2.1 静态库的制作

静态库使用 ar 指令进行打包:

ar -rc lib[库名].a [目标文件s]

lib[库名].a 是静态库文件的命名规范,实际上的库名需要去掉lib前缀以及.a扩展名。

通常来说,只有库文件是不够的,还需要将库的头文件交给用户,所以我们可以使用如下的Makefile来将库及其头文件一起打包交给用户:

SRC=$(wildcard *.c)
OBJ=$(SRC:.c=.o)

libmyc.a:$(OBJ)
	ar -rc $@ $^

$(OBJ):$(SRC)
	gcc -c $^

.PHONY:output
output:
	mkdir -p lib/include
	mkdir -p lib/mylib
	cp -f *.h lib/include
	cp -f *.a lib/mylib
	tar czf lib.tgz lib

.PHONY:clean
clean:
	rm -rf *.o libmyc.a lib lib.tgz

2.2 静态库的使用

gcc/g++会默认链接c标准库,但是myc库是我们自己制作的第三方库,所以在编译时需要指定链接myc库。

假设用户已经接收到了我们的 lib.tgz 包,并且用户的代码(usercode.c)调用了我们库中的方法:

注:使用tar xzf lib.tgz进行解包得到lib目录。 

// usercode.c

#include "mystdio.h"
#include "mystring.h"
#include <string.h>
#include <unistd.h>

int main()
{
    MyFile *filep = MyFopen("./log.txt", "a");
    if(!filep)
    {
        printf("fopen error!\n");
        return 1;
    }

    int cnt = 10;
    while(cnt--)
    {
        char *msg = (char*)"hello myfile!!!";
        MyFwrite(filep, msg, strlen(msg));
        MyFFlush(filep);
        printf("buffer: %s\n", filep->outbuffer);
        sleep(1);
    }
    MyFclose(filep); // FILE *fp

    const char *str = "hello bit!\n";
    printf("strlen: %d\n",my_strlen(str));
    return 0;
}
2.2.1 显式指定库文件及头文件路径

在编译时,需要指定头文件所在路径、要链接的库文件路径以及指定库文件:

gcc -o [可执行程序] [目标文件s] -I [头文件路径] -L [库路径] -l [库名]

2.2.2 将库文件安装到系统目录

我们知道,所谓安装,实际上就是把文件拷贝到指定的系统目录下。这样,在我们未显式指定库文件所在目录时,系统就能够在默认目录中找到。

当然,除了拷贝,建立链接也是可以的。 

  • /lib/usr/lib:系统级库
  • /usr/local/lib:用户安装的第三方库

我们将 libmyc.a 文件拷贝到三个库中的一个即可完成安装,此时不在需要指明库所在路径:

但是,不建议安装到系统级库,用户自己要安装的第三方库最好安装到 /usr/local/lib 中。 

2.2.3 将头文件安装到系统目录
  • /usr/include:系统级头文件
  • /usr/local/include:本地安装的第三方库头文件
  • /usr/include/<库名> 或 /usr/local/include/<库名>:特定软件的子目录

我们将自己的头文件拷贝到上述目录下即可完成安装,此时不再需要指明头文件所在路径:

3. 动态库

  • 文件扩展名:.so(Shared Object)

  • 特点:
    • 在程序运行时被动态加载到内存,多个程序可共享同一份库代码。

    • 可执行文件体积小,库更新时无需重新编译程序。

    • 缺点:依赖运行时环境中的库文件(若缺失会导致程序无法运行)。

  • 创建工具:gcc/g++ 的 -shared 选项。

  • 使用场景:大多数系统库(如 glibc)和通用功能库(如 OpenSSL)。

3.1 动态库的制作

// 编译目标文件时需要带上-fPIC选项,fPIC:产生位置无关码(position independent code) 
gcc/g++ -c -fPIC [原文件s]

// 生成库文件时需要带上-shared选项,shared: 表示生成共享库格式 
gcc/g++ -o lib[库名].so [目标文件s] -shared

同样的,lib[库名].so 是命名规范,实际上的库名需要去掉lib前缀和 .so扩展名。

我们可以使用如下的Makefile来对库及其头文件进行打包:

SRC=$(wildcard *.c)
OBJ=$(SRC:.c=.o)

libmyc.so:$(OBJ)
	gcc -shared -o $@ $^

$(OBJ):$(SRC)
	gcc -fPIC -c $^

.PHONY:output
output:
	mkdir -p lib/include
	mkdir -p lib/mylib
	cp -f *.h lib/include
	cp -f *.so lib/mylib
	tar czf lib.tgz lib

.PHONY:clean
clean:
	rm -rf *.o libmyc.so lib lib.tgz

3.2 动态库的使用

我们以同样的代码作为示例,将库及头文件安装到系统目录的方式与静态库一样,这里就不再重复,但是对于显式给出库文件路径的方式,我们要多说两句。

3.2.1 显式指定库文件路径

假如我们未将库文件安装到系统目录当中,并显式指定某路径下的库文件:

我们会发现编译通过了,但是:

当我们运行生成的可执行程序时,会发现系统显式找不到对应的库。

这是因为,我们仅仅告诉了编译器:“这个库是存在的”,所以编译器完成了编译。

但是动态链接是在程序运行时才将库与可执行程序产生链接,负责链接的是系统,然而系统并不知道在哪里找到这个库。 

要让操作系统在运行我们的程序时找到对应的动态库,我们可以选择安装的形式(与静态库的安装完全一致),也可采取以下几点中提到的措施。

 注意:与静态链接不同,接下来的几点措施(包括安装),都不需要重新编译可执行文件。

2.2.2 将路径加载到环境变量中 
# LD_LIBRARY_PATH:临时指定额外的库搜索路径。
export LD_LIBRARY_PATH=${LD_LIBRARY_PATH}:/path/to/libs

使用 ldd 命令可以查看可执行程序链接的库及其所在路径:

但是,这种方式只是临时的,重新登录或更新环境变量时就会失效。 

2.2.3 配置文件
  • /etc/ld.so.conf:系统级库路径配置文件
  • /etc/ld.so.conf.d:用户库路径配置文件目录

我们可以直接在/etc/ld.so.conf中加入我们库文件的路径,但是我们依然更建议在用户库路径配置文件目录中添加自己的配置文件:

这里sudo echo创建文件的方式居然不行,只能用编辑器创建了。

然后加载配置文件:

sudo ldconfig

 结果与2.2.2相同,这里就不展示了。

4. 总结与补充

  • gcc/g++编译命令补充:
    • [-I] :指定头文件所在目录。

    • [-L]:指定库文件所在路径。

    • [-l]:指定要链接的库。

    • [-shared]:生成动态库。

    • [-fPIC]:产生位置无关码。

    • [-static]:使用静态链接。

  • 静态库使用ar命令进行打包:
    ar -rc lib[库名].a [目标文件s]
  • 将静态库与用户目标文件一起编译即可生成可执行程序。
  •  动态库使用gcc/g++进行打包,且目标文件需要携带位置无关码:
    // 编译目标文件时需要带上-fPIC选项,fPIC:产生位置无关码(position independent code) 
    gcc/g++ -c -fPIC [原文件s]
    
    // 生成库文件时需要带上-shared选项,shared: 表示生成共享库格式 
    gcc/g++ -o lib[库名].so [目标文件s] -shared
  • 动态库在编译时需要让gcc/g++知道这个库是存在的(给出路径或安装到系统,并指定库名)。在运行时,系统需要能够找到这个库(需要安装到系统)。
  • 第三方库在编译时要指定链接这个库。
  • 在编译时,我们的系统当中可能既安装了某个库的动态版本,又安装了某个库的静态版本。此时,编译器默认能采用动态链接则采用动态链接。如果要使用静态链接则需要带上 -static 选项,一旦带上这个选项,就意味着动态链接被禁用,如果某个库只有动态链接的版本,则会发生链接失败。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

手机销售终端MPR+LTC项目项目总体方案P183(183页PPT)(文末有下载方式)

资料解读&#xff1a;手机销售终端 MPRLTC 项目项目总体方案 详细资料请看本解读文章的最后内容。在当今竞争激烈的市场环境下&#xff0c;企业的销售模式和流程对于其发展起着至关重要的作用。华为终端正处于销售模式转型的关键时期&#xff0c;波士顿 - 华为销售终端 MPRLTC …

【Python LeetCode Patterns】刷力扣,15 个学习模式总结

1. 前缀和&#xff08;Prefix Sum&#xff09;—— 查询子数组中元素和303. 区域和检索 - 数组不可变304. 二维区域和检索 - 矩阵不可变 2. 双指针&#xff08;Two Pointers&#xff09;—— 移向彼此或远离彼此3. 滑动窗口&#xff08;Sliding Window&#xff09;—— 找到满足…

蓝桥杯单片机刷题——串口发送显示

设计要求 通过串口接收字符控制数码管的显示&#xff0c;PC端发送字符A&#xff0c;数码管显示A&#xff0c;发送其它非法字符时&#xff0c;数码管显示E。 数码管显示格式如下&#xff1a; 备注&#xff1a; 单片机IRC振荡器频率设置为12MHz。 串口通信波特率&#xff1a;…

探索抓包利器ProxyPin,实现手机APP请求抓包,支持https请求

以下是ProxyPin的简单介绍&#xff1a; - ProxyPin是一个开源免费HTTP(S)流量捕获神器&#xff0c;支持 Windows、Mac、Android、IOS、Linux 全平台系统- 可以使用它来拦截、检查并重写HTTP(S)流量&#xff0c;支持捕获各种应用的网络请求。ProxyPin基于Flutter开发&#xff0…

文献学习:单细胞+临床+模型构建 | 一篇Molecular Cancer文献如何完整解读CDK4/6i耐药机制

&#x1f44b; 欢迎关注我的生信学习专栏~ 如果觉得文章有帮助&#xff0c;别忘了点赞、关注、评论&#xff01; &#x1f4cc;一、研究背景&#xff1a;CDK4/6i 是不是“万无一失”&#xff1f; HR/HER2- 是最常见的乳腺癌亚型&#xff0c;占比超过70%。近年来&#xff0c;随…

网盘解析工具更新,解决了一些bug

解析工具v1.2.1版本更新&#xff0c;本次是小版本更新&#xff0c;修复了一些bug。 之前小伙伴反应的网盘进入文件后不能返回上一级&#xff0c;现在这个bug修复了&#xff0c;已经可以点击了。 点击资源后会回到资源那一级目录&#xff0c;操作上是方便了不少。 增加了检查自…

5种生成模型(VAE、GAN、AR、Flow 和 Diffusion)的对比梳理 + 易懂讲解 + 代码实现

目录 1 变分自编码器&#xff08;VAE&#xff09;​ 1.1 概念 1.2 训练损失 1.3 VAE 的实现 2 生成对抗网络&#xff08;GAN&#xff09;​ 2.1 概念 2.2 训练损失 a. 判别器的损失函数 b. 生成器的损失函数 c. 对抗训练的动态过程 2.3 GAN 的实现 3 自回归模型&am…

计算机期刊推荐 | 计算机-人工智能、信息系统、理论和算法、软件工程、网络系统、图形学和多媒体, 工程技术-制造, 数学-数学跨学科应用

Computers, Materials & Continua 学科领域&#xff1a; 计算机-人工智能、信息系统、理论和算法、软件工程、网络系统、图形学和多媒体, 工程技术-制造, 数学-数学跨学科应用 期刊类型&#xff1a; SCI/SSCI/AHCI 收录数据库&#xff1a; SCI(SCIE),EI,Scopus,知网(CNK…

【教学类-58-14】黑白三角拼图12——单页1页图。参考图1页6张(黑白、彩色)、板式(无圆点、黑圆点、白圆点)、宫格2-10、张数6张,适合集体操作)

背景需求&#xff1a; 基于以下两个代码&#xff0c;设计一个单页1页黑白三角、彩色三角&#xff08;包含黑点、白点、无点&#xff09;的代码。 【教学类-58-12】黑白三角拼图10&#xff08;N张参考图1张操作卡多张彩色白块&#xff0c;适合个别化&#xff09;-CSDN博客文章…

C++项目:高并发内存池_下

目录 8. thread cache回收内存 9. central cache回收内存 10. page cache回收内存 11. 大于256KB的内存申请和释放 11.1 申请 11.2 释放 12. 使用定长内存池脱离使用new 13. 释放对象时优化成不传对象大小 14. 多线程环境下对比malloc测试 15. 调试和复杂问题的调试技…

消息队列性能比拼: Kafka vs RabbitMQ

本内容是对知名性能评测博主 Anton Putra Kafka vs RabbitMQ Performance 内容的翻译与整理, 有适当删减, 相关数据和结论以原作结论为准。 简介 在本视频中&#xff0c;我们将首先比较 Apache Kafka 和传统的 RabbitMQ。然后&#xff0c;在第二轮测试中&#xff0c;会将 Kaf…

AP 场景架构设计(一) :OceanBase 读写分离策略解析

说明&#xff1a;本文内容对应的是 OceanBase 社区版&#xff0c;架构部分不涉及企业版的仲裁副本功能。OceanBase社区版和企业版的能力区别详见&#xff1a; 官网链接。 概述​ 当两种类型的业务共同运行在同一个数据库集群上时&#xff0c;这对数据库的配置等条件提出了较高…

Java 大视界 -- Java 大数据在智能金融区块链跨境支付与结算中的应用(154)

&#x1f496;亲爱的朋友们&#xff0c;热烈欢迎来到 青云交的博客&#xff01;能与诸位在此相逢&#xff0c;我倍感荣幸。在这飞速更迭的时代&#xff0c;我们都渴望一方心灵净土&#xff0c;而 我的博客 正是这样温暖的所在。这里为你呈上趣味与实用兼具的知识&#xff0c;也…

手把手教你在linux服务器部署deepseek,打造专属自己的数据库知识库

第一步&#xff1a;安装Ollama 打开官方网址&#xff1a;https://ollama.com/download/linux 下载Ollama linux版本 复制命令到linux操作系统执行 [rootpostgresql ~]# curl -fsSL https://ollama.com/install.sh | sh在Service中增加下面两行 [rootlocalhost ~]# vi /etc/…

C++ 继承:面向对象编程的核心概念(一)

文章目录 引言1. 继承的基本知识1.1 继承的关键词的区别1.2 继承类模版 2. 基类和派生类间的转换3. 继承中的作用域4. 派生类的默认成员函数4.1 默认成员函数的规则4.2 自己实现成员函数4.3 实现一个不能被继承的基类&#xff08;基本不用&#xff09; 引言 在C中&#xff0c;…

蓝桥杯 临时抱佛脚 之 二分答案法与相关题目

二分答案法&#xff08;利用二分法查找区间的左右端点&#xff09; &#xff08;1&#xff09;估计 最终答案可能得范围 是什么 &#xff08;2&#xff09;分析 问题的答案 和 给定条件 之间的单调性&#xff0c;大部分时候只需要用到 自然智慧 &#xff08;3&#xff09;建…

【算法day22】两数相除——给你两个整数,被除数 dividend 和除数 divisor。将两数相除,要求 不使用 乘法、除法和取余运算。

29. 两数相除 给你两个整数&#xff0c;被除数 dividend 和除数 divisor。将两数相除&#xff0c;要求 不使用 乘法、除法和取余运算。 整数除法应该向零截断&#xff0c;也就是截去&#xff08;truncate&#xff09;其小数部分。例如&#xff0c;8.345 将被截断为 8 &#x…

关于服务器只能访问localhost:8111地址,局域网不能访问的问题

一、问题来源&#xff1a; 服务器是使用的阿里云的服务器&#xff0c;服务器端的8111端口没有设置任何别的限制&#xff0c;但是在阿里云服务器端并没有设置相应的tcp连接8111端口。 二、解决办法&#xff1a; 1、使用阿里云初始化好的端口&#xff1b;2、配置新的阿里云端口…

基于ADMM无穷范数检测算法的MIMO通信系统信号检测MATLAB仿真,对比ML,MMSE,ZF以及LAMA

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 4.1 ADMM算法 4.2 最大似然ML检测算法 4.3 最小均方误差&#xff08;MMSE&#xff09;检测算法 4.4 迫零&#xff08;ZF&#xff09;检测算法 4.5 OCD_MMSE 检测算法 4.6 LAMA检测算法 …

Linux 配置时间服务器

一、同步阿里云服务器时间 服务端设置 1.检查chrony服务是否安装&#xff0c;设置chrony开机自启&#xff0c;查看chrony服务状态 [rootnode1-server ~]# rpm -q chrony # rpm -q 用于查看包是否安装 chrony-4.3-1.el9.x86_64 [rootnode1-server ~]# systemctl enable --n…