静态库(.a)与动态库(.so)的简明介绍
静态库(.a)与动态库(.so)的简明介绍
gcc有很多关于静态库,动态库的选项如-l
,-L
,-fPIC
,-shared
-Wl,-soname
,看着很复杂容易混淆,其实静态库和动态库都是应需而生,只要有了一个线索都很容易理解。
普通编译
假设有三个文件(后面均使用这个例子):
// mod1.c
#include <stdio.h>
void print_mod1(){
printf("%s\n",__func__);
}
// mod2.c
#include <stdio.h>
void print_mod2(){
printf("%s\n",__func__);
}
//main.c
int main(){
print_mod1();
print_mod2();
return 0;
}
我们要想运行这个程序需要先编译mod1.c
,mod2.c
生成目标文件,然后目标文件与main.c
结合完成编译:
$ gcc -c mod1.c mod2.c
$ gcc -o resultant main.c mod1.o mod2.o
$ ./resultant
print_mod1
print_mod2
使用静态库(-lname -Lpath)
如果mod再多一点就会出现体力劳动:
$ gcc -o resultant mod1.o mod2.o mod3.o mod4.o mod5.o ... mod1024.o
于是就引入了静态库的概念。静态库又叫归档文件,在linux下是*.a
后缀的文件,本质上就是目标文件(*.o
)的一个集合。
- 使用
ar -r
命令可以将*.o
打包为一个静态库
$ ar r libmod.a mod1.o mod2.o
- 使用
ar tv libmod.a
查看归档里面有哪些目标文件:
$ ar tv libmod.a
rw-r--r-- 0/0 1544 Dec 31 16:00 1969 mod1.o
rw-r--r-- 0/0 1544 Dec 31 16:00 1969 mod2.o
- 使用
ar d libmod.a mod2.o
删除一个不需要的目标文件
打包好后就可以用libmod.a
代替一串目标文件了:
$ gcc -o resultant main.c libmod.a
最常用的链接静态库的方式是添加-lname
选项。-lname
会默认链接名为libname.a
的静态库:
$ gcc -o resultant main.o -lmod
/usr/bin/ld: cannot find -lmod
collect2: error: ld returned 1 exit status
这里我们如果直接使用-lmod
gcc会提示找不到libmod.a
模块,因为gcc只会在标准路径如/usr/lib,/lib
查找,解决方法一是把libmod.a放到标准路径,二是使用-Lpath
选项。
$ gcc -o resultant main.c -L. -lmod
$ ./resultant
print_mod1
print_mod2
-Lpath
把指定路径加入链接器搜索路径,这里我们把当前目录(.
)加入,自然就能找到libmod.a
了。
使用动态库(-fPIC -shared)
静态库优点是方便,缺点是每个程序都有一份目标文件的,很多程序会使用printf
,如果每个程序都包含一份printf.o
的实现,会非常浪费磁盘空间和宝贵内存页。还有如果要对静态库中某一个目标文件进行更新,那么应用程序就需要重新链接。
基于这些需求,动态库诞生了。
动态库需要位置独立的代码,所以不能使用前面的mod1.o
,mod2.o
,需要-fPIC
选项重新编译:
$ gcc -c -fPIC mod1.c mod2.c
然后再组合成动态库:
$ gcc -shared -o libmod.so mod1.o mod2.o
最后使用这个动态库:
$ gcc -o resultant main.c libmod.so
$ ./resultant
./resultant: error while loading shared libraries: libmod.so: cannot open shared object file: No such file or directory
好了,不出所料,又出问题了。gcc提示加载动态库失败,找不到它。动态库的搜索顺序如下
- 编译目标代码时指定的动态库搜索路径;
- 环境变量LD_LIBRARY_PATH指定动态库搜索路径,它指定程序动态链接库文件搜索路径;export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:data/home/billchen/lib
- 配置文件/etc/ld.so.conf中指定的动态库搜索路径;
- 默认的动态库搜索路径/lib;
- 默认的动态库搜索路径/usr/lib。
这里简单起见,我们直接将libmod.so
移动到/lib
:
$ sudo mv libmod.so /lib
$ ./resultant
print_mod1
print_mod2
动态库别名(-Wl,-soname,xx)
这里再说说-Wl,-soname
,该选择告知链接器一个动态库的别名
$ gcc -shared -Wl,-soname,libalias.so -o libmod.so mod1.o mod2.o
$ gcc -o resultant main.c libmod.so
上面命令使用libalias.so
作为libmod.so
的别名,再次运行resultant
会提示找不到libalias.so
错误而不是libmod.so
,-soname别名引入一个中间层,好处是程序运行时可以使用和编译时不一样的兼容库。
静态库(.a)与动态库(.so)的简明介绍的更多相关文章
- [转]Linux下g++编译与使用静态库(.a)和动态库(.os) (+修正与解释)
在windows环境下,我们通常在IDE如VS的工程中开发C++项目,对于生成和使用静态库(*.lib)与动态库(*.dll)可能都已经比较熟悉,但是,在linux环境下,则是另一套模式,对应的静态库 ...
- 《CMake实践》笔记三:构建静态库(.a) 与 动态库(.so) 及 如何使用外部共享库和头文件
<CMake实践>笔记一:PROJECT/MESSAGE/ADD_EXECUTABLE <CMake实践>笔记二:INSTALL/CMAKE_INSTALL_PREFIX &l ...
- 在Linux中创建静态库.a和动态库.so
转自:http://www.cnblogs.com/laojie4321/archive/2012/03/28/2421056.html 在Linux中创建静态库.a和动态库.so 我们通常把一些公用 ...
- windows动态库与Linux动态库
Linux动态库和windows动态库的目的是基本一致的,但由于操作系统的不同,他们在许多方面还是不尽相同.但是尽管有差异Linux动态库的windows动态库还是可以移植的,有一些规则以及经验是必须 ...
- Linux C 静态库(.a) 与 动态库(.so) 的详解
库从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行.库分静态库和动态库两种. 一.静态库和动态库的区别 1.静态函数库 这类库的名字一般是libxxx.a:利用静态函数库编译成的文件比较 ...
- [付费视频]Delphi视频Android开发使用静态库(A)和动态库(SO)
关于本视频:前阵子接到一个委托,解决Delphi开发Android程序中串口通信的问题,厂家那边提供了c文件,需要翻译成delphi可用,翻译倒是比较简单.不过后来翻译读写ic卡单元的时候进行不下去了 ...
- Linux链接库二(动态库,静态库,库命名规则,建立个没有版本号的软连接文件)
http://www.cppblog.com/wolf/articles/74928.html http://www.cppblog.com/wolf/articles/77828.html http ...
- Linux链接库一(动态库,静态库,库放在什么路径下)
http://www.cppblog.com/wolf/articles/74928.html http://www.cppblog.com/wolf/articles/77828.html http ...
- C/C++ 关于生成静态库(lib)/动态库(dll)文件如何使用(基于windows基础篇)
1. 首先,如何制作一个静态库(lib)? 额, 对于静态库,我们知道,里头是不应该有Main函数,它只是一个配合文件.之所以称之为lib静态库,其实就是指,我们需要用到lib里头的函数时,我们才会去 ...
随机推荐
- c++builder PM2.5
c++builder PM2.5 TMemoryStream *ms = new TMemoryStream(); this->NetHTTPClient1->Get("http ...
- Python基础学习五 内置函数
1.函数补充: 1)函数返回值return可以有多个 2)补充示例: nums = [0,1,2,3,4,5,6,7,8] #如何将list里面的元素变为字符串类型 new_nums = [str(x ...
- 嵌入式Linux启动配置文件及脚本分…
使用Busybox制作根文件系统时,/etc目录非常重要,它包含了嵌入式Linux启动所需的配置文件及脚本.由于init进程,或者说linuxrc程序会解析inittab文件,因此就从/etc/ini ...
- (转)Mac下MySql安装经历(含安装错误排查、卸载多种折腾)
在安装mysql的时候,活活折腾我两天.结果终于被我折腾成功了……一开始我就放了个错误:我下了32位版本的mysql:mysql-5.5.8-osx10.6-x86.dmg 须知在mac下装的是64位 ...
- mfs权威指南
1. 我在性能测试中间遇到些问题,因为我时间有限,所以希望大家一起来测试解决,群策群力.有什么问题请大家及时指出来,因为我也处在一个不断摸索的阶段. 2. mfs不多做介绍,具体细节请参考本版mfs实 ...
- 流Stream
System.IO 提供了一个抽象类Stream , Stream类 支持对字节的读写操作.所谓的“流”,指的是Stream,也就是所谓的一个文件区.这个文件区中存储着的信息可以是在内存中,也可以是在 ...
- 680. Valid Palindrome II 对称字符串-可删字母版本
[抄题]: Given a non-empty string s, you may delete at most one character. Judge whether you can make i ...
- 17-取石子-hdu1846(巴什博奕)
http://acm.hdu.edu.cn/showproblem.php?pid=1846 Brave Game Time Limit: 1000/1000 MS (Java/Others) ...
- 面试题:servlet jsp cook session 背1
一.Servlet是什么?JSP是什么?它们的联系与区别是什么? Servlet是Java编写的运行在Servlet容器的服务端程序,狭义的Servlet是指Servlet接口,广义的Servlet是 ...
- 如何安全退出已调用多个Activity的应用
对于单一Activity的应用来说,退出很简单,直接finish()即可.当然,也可以用killProcess()和System.exit()这样的方法. 但是,对于多Activity的应用来说,在打 ...