Linux下库函数、静态库与动态库

库函数

什么是库

   库是二进制文件, 是源代码文件的另一种表现形式, 是加了密的源代码;

是一些功能相近或者是相似的函数的集合体.

使用库有什么好处

  • 提高代码的可重用性, 而且还可以提高程序的健壮性;
  • 可以减少开发者的代码开发量, 缩短开发周期.

库制作完成后, 如何给用户使用

  • 头文件---包含了库函数的声明
  • 库文件---包含了库函数的代码实现

注意: 库不能单独使用, 只能作为其他执行程序的一部分完成某些功能,

就是说只能被其他程序调用才能使用.

库可分静态库(static library)和共享库(shared library)

代码:

静态库

静态库可以认为是一些目标代码的集合, 是在可执行程序运行前就已经加入到执行码中, 成为执行程序的一部分. 按照习惯, 一般以.a做为文件后缀名.

静态库的命名一般分为三个部分:

  • 前缀:lib
  • 库名称:自定义即可, test
  • 后缀:.a

所以最终的静态库的名字应该为:libtest.a

静态库的制作

下面以fun1.c , fun2.chead.h三个文件为例讲述静态库的制作和使用, 其中head.h文件中有函数的声明,  fun1.cfun2.c中有函数的实现.

步骤1:将c源文件生成对应的.o文件

                 gcc -c fun1.c fun2.c

   或者分别生成.o文件:

                gcc -c fun1.c -o fun1.o

                gcc -c fun2.c -o fun2.o

步骤2:使用打包工具ar将准备好的.o文件打包为.a文件

  1. 在使用ar工具是时候需要添加参数rcs
  • r更新、c创建、s建立索引
  1. 命令:ar rcs 静态库名 .o文件
  • ar rcs libtest1.a fun1.o fun2.o

静态库的使用

静态库制作完成之后, 需要将.a文件和头文件一定发布给用户.

假设测试文件为main.c, 静态库文件为libtest1.a, 头文件为head.h

用到的参数:

  • -L:指定要连接的库的所在目录
  • -l:指定链接时需要的静态库, 去掉前缀和后缀(这是小写L)
  • -I: 指定main.c文件用到的头文件head.h所在的路径(这是大写i)

gcc -o main1 main.c -L./ -ltest1 -I./

静态库的优缺点

  1. 优点:

    • 函数库最终被打包到应用程序中,实现是函数本地化,寻址方便、速度快。

(库函数调用效率==自定义函数使用效率)

    • 程序在运行时与函数库再无瓜葛,移植方便。
  1. 缺点:

    • 消耗系统资源较大, 每个进程使用静态库都要复制一份, 无端浪费内存。
    • 静态库会给程序的更新、部署和发布带来麻烦。如果静态库libxxx.a更新了,所有使用它的应用程序都需要重新编译、发布给用户(对于玩家来说,可能是一个很小的改动,却导致整个程序重新下载)。

代码:

动态库

共享库在程序编译时并不会被连接到目标代码中, 而是在程序运行是才被载入. 不同的应用程序如果调用相同的库, 那么在内存里只需要有一份该共享库的拷贝, 规避了空间浪费问题.动态库在程序运行时才被载入, 也解决了静态库对程序的更新、部署和发布会带来麻烦. 用户只需要更新动态库即可, 增量更新. 为什么需要动态库, 其实也是静态库的特点导致.

按照习惯, 一般以”.so”做为文件后缀名. 共享库的命名一般分为三个部分:

  • 前缀:lib
  • 库名称:自己定义即可, test
  • 后缀:.so

所以最终的静态库的名字应该为:libtest.so

共享库的制作

  1. 生成目标文件.o, 此时要加编译选项:-fPICfpic

gcc -fpic -c fun1.c fun2.c

参数:-fpic创建与地址无关的编译程序(pic, position independent code), 目的就是为了能够在多个应用程序间共享.

  1. 生成共享库, 此时要加链接器选项: -shared(指定生成动态链接库

gcc -shared fun1.o fun2.o -o libtest2.so

共享库的使用

引用动态库编译成可执行文件(跟静态库方式一样):

用到的参数:

  • -L:指定要连接的库的所在目录
  • -l:指定链接时需要的动态库, 去掉前缀和后缀
  • -I: 指定main.c文件用到的头文件head.h所在的路径

gcc main.c -I./ -L./ -ltest2 -o main2

然后运行:./main2,发现竟然报错了.

分析为什么在执行的时候找不到libtest2.so

  • 当系统加载可执行代码时候, 能够知道其所依赖的库的名字, 但是还需要知道所依赖的库的绝对路径。此时就需要系统动态载入器(dynamic linker/loader)。

ldd命令可以查看可执行文件依赖的库文件, 执行ldd main2, 可以发现libtest2.so找不到.

  • 对于elf格式的可执行程序,是由ld-linux.so*来完成的, 它先后搜索elf文件的 DT_RPATH环境变量LD_LIBRARY_PATH/etc/ld.so.cache文件列表/lib/, /usr/lib目录找到库文件后将其载入内存。

使用file命令可以查看文件的类型: file main2

如何让系统找到共享库

  • 拷贝自己制作的共享库到/lib或者/usr/lib
  • 临时设置LD_LIBRARY_PATH:
    • export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径
  • 永久设置, export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径, 设置到/.bashrc文件中, 然后在执行下列三种办法之一:
  • 执行. ~/.bashrc使配置文件生效(第一个.后面有一个空格)
  • 执行source ~/.bashrc配置文件生效
  • 退出当前终端, 然后再次登陆也可以使配置文件生效
  • 永久设置,把export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径,设置到/etc/profile文件中
  • 将其添加到 /etc/ld.so.cache文件中
    • 编辑/etc/ld.so.conf文件, 加入库文件所在目录的路径
    • 运行sudo ldconfig -v, 该命令会重建/etc/ld.so.cache文件

解决了库的路径问题之后, 再次ldd命令可以查看可执行文件依赖的库文件, ldd main2:

共享库的特点

  • 动态库把对一些库函数的链接载入推迟到程序运行的时期。
  • 可以实现进程之间的资源共享。(因此动态库也称为共享库)
  • 将一些程序升级变得简单。
  • 甚至可以真正做到链接载入完全由程序员在程序代码中控制(显示调用)

代码:

文件夹所属情况:

修改动态库的路径:

改完之后一切按步骤进行即可!

我设置的是工作文件中的lib,而不是用户目录下的lib

那为啥要将库函数打包了?

我gcc add.c main.c -o main不照样可以执行?

将库函数打包成库的主要目的之一是为了代码的重用。当您有多个程序需要使用相同的函数或者模块时,将这些函数或模块打包成库可以节省时间和精力,并提高代码的可维护性。库函数可以被多个程序共享,而不需要每个程序都重新编写一遍。

另一个重要的原因是为了减少可执行文件的大小。如果您使用静态库,编译器会将库函数的代码和数据嵌入到可执行文件中,这会增加可执行文件的大小。而如果使用动态库,可执行文件只包含对动态库的引用,因此可执行文件的大小会更小。这对于大型项目或需要频繁分发的程序来说尤为重要。

库函数与静态库、动态库的关系:

将add.o和sub.o打包成静态库后,就是将文件中的数据复制到静态库文件中

静态库

在创建静态库时,这些目标文件会被归档到静态库中,使得静态库中包含了这些目标文件中的所有代码和数据。这样,当链接器在编译时将静态库链接到可执行文件时,它会从静态库中提取所需的代码和数据,然后将其合并到最终的可执行文件中。

动态库

动态库与静态库类似,也包含了代码和数据,但与静态库不同的是,动态库在运行时由动态链接器加载到内存中。因此,动态库文件中包含了一些额外的信息,以支持动态链接器在运行时正确地加载和链接动态库。

动态库文件的格式通常比较复杂,包含了符号表、重定位表、动态链接信息等。这些信息使得动态链接器能够在程序运行时正确地加载和链接动态库,并且支持库的版本管理和符号重定位等功能。

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

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

相关文章

YUM | 包安装 | 管理

YUM 功能 软件包安装&#xff1a; 通过yum命令安装软件包。例如&#xff0c;安装一个名为 example-package 的软件包 yum install example-package更新包 检查更新&#xff1a; 检查可用更新&#xff1a; sudo yum check-update <package_name>软件包更新&#xff1a; y…

Win32 SDK Gui编程系列之--ListView自绘OwnerDraw

ListView自绘OwnerDraw 1.ListView自绘OwnerDraw 正在试错是否使用了列表视图,尽量制作出智能的表格编辑器。本页显示了业主抽签的表格数据(二维数组数据)的显示方法。 显示画面和整个程序如下所示。使用ListView_GetSubItemRect宏的话,就不需要getRect函数了。 当nCol的…

Linux基础开发工具使用【Yum | Vim | 编译器 | 调试器 | GDB | Makefile | Git】

本节目标&#xff1a; 学习yum工具&#xff0c;进行软件安装掌握vim编辑器使用&#xff0c;学会vim的简单配置掌握简单的Makefile编写&#xff0c;了解其运行思想编写自己的第一个Linux 程序&#xff1a;进度条掌握gcc/g编译器的使用&#xff0c;并了解其过程&#xff0c;原理掌…

蓝桥杯-求阶乘-python

问题描述 满足N!的末尾恰好有K个0的最小的N是多少&#xff1f; 如果这样的N不存在输出一1。 思路解析 末尾的0是由10产生的&#xff0c;而10是由质数2和5产生的 在求阶乘的过程中&#xff0c;只要是偶数就会有2&#xff0c;而5相对2更少&#xff0c;所以对于10的数量我们可以…

Python程序员面试题精选(1)

本文精心挑选了10道Python程序员面试题&#xff0c;覆盖了Python的多个核心领域&#xff0c;包括装饰器、lambda函数、列表推导式、生成器、全局解释器锁(GIL)、单例模式以及上下文管理器等。每道题都附有简洁的代码示例&#xff0c;帮助读者更好地理解和应用相关知识点。 题目…

redis:七、集群方案(主从复制、哨兵模式、分片集群)和面试模板

redis集群方案 在Redis中提供的集群方案总共有三种&#xff08;一般一个redis节点不超过10G内存&#xff09; 主从复制哨兵模式分片集群 主从复制&#xff08;主从数据同步&#xff09; replid和offset Replication Id&#xff1a;简称replid&#xff0c;是数据集的标记&a…

离谱!英国大学53%本科生用AI写论文!留学生该如何面对AI冲击?

随着AI的不断强化和更新换代&#xff0c;越来越多的学生开始使用人工智能辅助写论文&#xff0c;能省去不少信息搜集的时间。 英国大学从最开始的明令禁止&#xff0c;到如今也在逐步接受学生将它用做辅助工具&#xff1a; 然而&#xff0c;AI的使用其实还存在很多问题&#xf…

Files的常用方法都有哪些

Files的常用方法都有哪些 Java的java.nio.file包提供了Files工具类&#xff0c;用于对文件和目录进行常见的操作。以下是一些Files类的常用方法&#xff0c;以及简单的代码演示&#xff1a; 复制文件或目录&#xff1a; import java.io.IOException; import java.nio.file.*;pu…

寒假 day6

1.请编程实现二又树的操作。 1.1二又树的创建 1.2二又树的先序遍历 1.3二又树的中序遍历 1.4二又树的后序遍历 1.5二又树各个节点度的个数 1.6二叉树的深度 #include<stdio.h> #include<string.h> #include<stdlib.h> typedef char datatype; //定义节点结…

隧道穿透:端口转发、socket隧道代理

目录 端口转发 lcx工具 Lcx工具正向连接 Lcx工具反向连接 SOCKET隧道代理 socks常见利用场景 Proxifier SocksCap64 Proxychains 端口转发 本篇会和搭建介绍一下端口转发和socket隧道代理的概念和简单演示 lcx工具 lcx工具是一个红队人员在内网渗透测试中最典型的端…

【Django】Cookie和Session的使用

Cookies和Session 1. 会话 从打开浏览器访问一个网站&#xff0c;到关闭浏览器结束此次访问&#xff0c;称之为一次会话。 HTTP协议是无状态的&#xff0c;导致会话状态难以保持。 Cookies和Session就是为了保持会话状态而诞生的两个存储技术。 2. Cookies 2.1 Cookies定…

SpringCloud--Gateway解析

一、Gateway简介 Gateway是Spring Cloud官方推出的第二代微服务网关&#xff0c;它旨在提供统一的路由方式以及为微服务应用提供强大的负载均衡能力。与第一代Spring Cloud Netflix Zuul相比&#xff0c;Spring Cloud Gateway在性能、可扩展性、易用性等方面都有了显著的提升。…