Linux下静态链接库与动态链接库的区别

Linux下静态链接库与动态链接库的区别

引言

通常情况下,对函数库的链接是放在编译时期(compile time)完成的。所有相关的对象文件 (object file)与牵涉到的函数库(library)被链接合成一个可执行文件 (executable file)。程序运行 时,与函数库再无瓜葛,因为所有需要的函数已拷贝到自己门下。所以这些函数库被成为静态库(static libaray),通常文件 名为“libxxx.a”的形式。

其实,我们也可以把对一些库函数的链接载入推迟到程序运行的时期(runtime)。这就是如雷贯耳的动态链接库(dynamic link library)技术。


一 例子详解

文件目录树如下:

   1. libtest/ 

   2. |-- myjob.c 

   3. |-- myjob.h 

   4. |-- test.c 

静态库

A.做成静态库  libmyjob.a 

1. $ gcc  -c  myjob.c    -o  myjob.o 

2. $ ar  -c -r -s  libmyjob.a  myjob.o

B.链接

   1. $ gcc  test.o  libmyjob.a  -o  test 

C.引用库情况(无所要信息)

   1. $ ldd test 

   2. linux-gate.so.1 => (0xffffe000) 

   3. libc.so.6 => /lib/libc.so.6 (0xb7e29000) 

   4. /lib/ld-linux.so.2 (0xb7f6e000) 

动态库

A.做成动态库  libmyjob.so 

1. $ gcc  -Wall –fPIC    -c  myjob.c    -o  myjob.o   

  2. $ gcc -shared -o libmyjob.so myjob.o   

-shared:  该选项指定生成动态连接库(让连接器生成T 类型的导出符号表,有时候也生成弱连接 W 类型的导出符号),不用该标志外部程序无法连接。相当于一个可执行文件。

-fPIC 表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。

-L. 表示要连接的库在当前目录中。

LD_LIBRARY_PATH 这个环境变量指示动态连接器可以装载动态库的路径。

B.链接

链接方法I ,拷贝到系统库里再链接,让 gcc 自己查找:

   1. $ cp libmyjob.so /usr/lib 

   2. $ gcc -o test test.o -lmyjob 

这里我们可以看到了 -lmyjob  选项, -l[lib_name]  指定库名,他会主动搜索。 lib[lib_name].so 这个搜索的路径可以通过  gcc --print-search-dirs 来查找。

链接方法II ,手动指定库路径

   1. $  gcc -o test test.o -lmyjob -B /path/to/lib

-B  选项就添加 /path/to/lib  gcc 搜索的路径之中。这样链接没有问题但是方法 II 中手动链接好的程序在 执行 时候仍旧需要指定库路径( 链接和执行是分开的 )。需要添加系统变量 LD_LIBRARY_PATH :

   1. $ export LD_LIBRARY_PATH=/path/to/lib 

这个时候再来检测一下test 程序的库链接状况 ( 方法 I 情况 )

   1. $ ldd test 

   2. linux-gate.so.1 => (0xffffe000) 

   3. libmyjob.so => /usr/lib/ libmyjob .so (0xb7f58000) 

   4. libc.so.6 => /lib/libc.so.6 (0xb7e28000) 

   5. /lib/ld-linux.so.2 (0xb7f6f000) 

 是不是比静态链接的程序多了一个 libmyjob.so?  这就是静态与动态的最大区别,静态情况下,它把库直接加载到程序里,而在动态链接的时候,它只是保留接口,将动态库与程序代码独立。这样就可以提高代码的可复用度,和降低程序的耦合度。

    另外,运行时,要保证主程序能找到动态库,所以动态库一般发布到系统目录中,要么就在跟主程序相对很固定的路径里,这样不管主程序在本机何时何地跑,都能找得到动态库。而静态库只作用于链接时,运行主程序时静态库文件没存在意义了。


二 静态库和动态库的区别

1. 静态函数库 

这类库的名字一般是 libxxx.a ;利用静态函数库编译成的文件比较大,因为整个 函数库的所有数据都会被整合进目标代码中,他的优点就显而易见了,即编译后的执行程序不需要外部的函数库支持,因为所有使用的函数都已经被编译进去了。当然这也会成为他的缺点,因为 如果静态函数库改变了,那么你的程序必须重新编译 。 

2. 动态函数库 

这类库的名字一般是 libxxx.so ;相对于静态函数库,动态函数库在编译的时候并没有被编译进目标代码中,你的程序执行到相关函数时才调用该函数库里的相应函数,因此动态函数库所产生的可执行文件比较小。由于函数库没有被整合进你的程序,而是程序运行时动态的申请并调用,所以程序的运行环境中必须提供相应的库。 动态函数库的改变并不影响你的程序,所以动态函数库的升级比较方便。

 

linux静态函数库的创建和使用 

A. 例程 str_out.h str_out.c main.c

/* str_out.h */

#ifndef STR_OUT_H

#define STR_OUT_H

void str_out(const char* str);

#endif

/* str_out.c */

#include "str_out.h"

void str_out(const char* str)

{

    printf("%s / n",str);

}

/* main.c */

int main()

{

    str_out("myjob world");

    return 0;

(第一步 ) gcc -c str_out.c  - o  str_out.o

B.静态函数库由 ar 命令创建 

(第二步 ) ar crs libstr_out.a str_out.o

-c : create 的意思 

-r :replace 的意思,表示当插入的模块名已经在库中存在,则替换同名的模块。如果若干模块中有一个模块在库中不存在,ar 显示一个错误消息,并不替换其他同名模块。


-s: 代表若归档文件中包含了对象模式(C++)

C. 使用方法

(第三步 ) gcc  main.c -o out -L. -lstr_out

通过gcc -o out main.c -L. -lstr_out 编译 main.c 就会把静态函数库整合进 out 。 

-L: 指定静态函数库的位置供查找,注意L 后面还有 '.' ,表示静态函数库在本目录下查找。 

-l: 则指定了静态函数库名,由于静态函数库的命名方式是lib***.a ,其中的 lib .a 忽略。 

根据静态函数库的特性,此处删除libstr_out.a out 依然可以运行,因为静态库的内容已经整合进去了。 

动态函数库的创建和使用 

A. 创建动态库  

( 第一步 ) gcc -fPIC -Wall -c str_out.c  o str_out.o

(第二步 ) gcc -shared -Wl, -soname,libstr_out.so.1 -o libstr_out.so str_out.o  

该命令生成libstr_out.so  动态函数库。 

-shared 指定生成动态链接库。 

-static 指定生成静态链接库。 

-fPIC 表示编译为位置独立的代码,用于编译共享库。目标文件需要创建成位置无关码,概念上就是在可执行程序装载它们的时候,它们可以放在可执行程序的内存里的任何地方。 

-L. 表示要连接的库在当前目录中。 

-l 指定链接时需要的动态库。编译器查找动态连接库时有隐含的命名规则,即在给出的名字前面加上lib ,后面加上 .so 来确定库的名称。 

-Wl,options 把参数(options) 传递给链接器 ld 。如果 options 中间有逗号 , 就将 options 分成多个选项 , 然后传递给链接程序。

B.动态库的使用

这时还不能立即./out ,因为在动态函数库使用时,会查找 /usr/lib/lib 目录下的动态函数库,而此时我们生成的库不在里边。 

1.最简单的方法就是把 libstr_out.so 拉到 /usr/lib /lib 中去。

(第三步 ) gcc main.c -o main -lstr_out

2. export LD_LIBRARY_PATH=$(pwd) 

    (第三步 gcc  main.c - o  main   -L.   -lstr_out

3. bashrc profile 文件里用 LD_LIBRARY_PATH 定义,然后用 source 加载。

4.还可以在 /etc/ld.so.conf 文件里加入我们生成的库的目录,然后 /sbin/ldconfig 。 

/etc/ld.so.conf是非常重要的一个目录,里面存放的是链接器和加载器搜索共享库时要检查的目录,默认是从 /usr/lib  /lib 中读取的 为了让动态链接库为系统所共享,还需运行动态链接库的管理命令--ldconfig.此执行程序存放在/sbin目录下.ldconfig命令的用途,主要是在默认搜寻目录(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所列的目录下,搜索出可共享的动态链接库(格式如前介绍,lib*.so*),进而创建出动态装入程序(ld.so)所需的连接和缓存文件.缓存文件默认为 /etc/ld.so.cache,此文件保存已排好序的动态链接库名字列表.ldconfig通常在系统启动时运行,而当用户安装了一个新的动态链接库时,就需要手工运行这个命令.

 

 

另外还有个文件需要了解/etc/ld.so.cache, 里面保存了常用的动态函数库,且会先把他们加载到内存中,因为内存的访问速度远远大于硬盘的访问速度,这样可以提高软件加载动态函数库的速度了。

    注意:默认情况下使用-l选项将搜索动态链接库(.so),没有对应.so文件时将使用.a文件进行静态编译。

  • 5
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: Linux动态链接库是一种在程序运行时动态加载的库文件,它包含了一些可被程序调用的函数和变量。与静态链接库不同,动态链接库在程序运行时才会被加载,可以减小程序的体积,提高程序的灵活性和可移植性。动态链接库可以被多个程序共享,从而节省系统资源。在Linux系统中,动态链接库的文件名通常以“.so”结尾。 ### 回答2: Linux动态链接库是一种在运行时加载的共享库,它可以被多个程序共享使用,提供可重用的代码和功能。与静态链接库相比,动态链接库允许程序在运行时加载,并在内存中共享已加载的库。这种方式可以减少内存占用,提高系统性能,并降低可执行文件的大小。 Linux动态链接库分为两种类型:系统级动态链接库和用户级动态链接库。系统级动态链接库例如libc,libpthread等,它们提供了操作系统的基本功能和系统调用的封装。用户级动态链接库则是由开发者自行编写的共享库,它包含了特定功能的代码,可以被多个程序调用和重用。 动态链接库的使用有以下优势: 1. 代码重用:多个程序可以共享同一个动态链接库,避免了重复代码的编写和维护。 2. 系统升级:当动态链接库的代码需要更新时,只需要替换库文件,而不需要重新编译所有使用该库的程序。 3. 空间节省:动态链接库在内存中只需加载一次,并被多个程序共享使用,减少了内存占用和可执行文件的大小。 4. 运行时动态性:程序可以在运行时加载或卸载动态链接库,增加了程序的灵活性和扩展性。 然而,使用动态链接库也存在一些注意事项: 1. 依赖关系:程序需要正确的动态链接库版本,否则会导致运行错误或崩溃。 2. 性能开销:动态链接库的加载和链接需要额外的时间和开销,可能会稍微降低程序的执行速度。 3. 安全性:动态链接库易受到恶意代码的攻击,因此需要确保库的来源和完整性。 总之,Linux动态链接库是一种强大的机制,可以提高代码的重用性和灵活性,在系统开发和维护中起到了重要的作用。 ### 回答3: Linux动态链接库(Dynamic Link Library,简称DLL)是指一种在Linux操作系统中,将不同的代码库以动态链接的方式组合在一起,供程序调用的机制。 在Linux中,动态链接库是以共享库(shared library)的形式存在的。它与静态链接库(static library)相比,可以在程序运行时动态加载和卸载,并且可以被多个程序同时共享使用。 Linux动态链接库的特点有以下几点: 1. 节省内存空间:相比静态链接库动态链接库只需在内存中加载一次,就可以被多个程序共享使用,节省了内存空间。 2. 灵活更新:由于动态链接库是以单独的文件存在的,所以可以通过替换或更新动态链接库文件,实现对其功能的升级或修复,而不用重新编译和链接整个程序。 3. 高度可移植性:动态链接库可以跨平台使用,只需在目标系统中存在相同的动态链接库文件即可。 4. 动态加载和卸载:在程序运行时,可以通过动态链接库的加载和卸载机制,动态添加或删除某个功能模块,实现程序的动态扩展和精简。 5. 提高开发效率:动态链接库将一些常用的函数和代码库进行打包,供多个程序调用,可以提高开发效率,并且减少了代码重复性。 总之,Linux动态链接库是一种方便、灵活、可共享的代码组织方式,提高了程序的开发效率和可维护性,也为程序的升级和优化提供了便利。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值