C/C++ GCC/Clang生成和链接静态库/动态库
{
//http://www.360doc.com/content/14/0413/16/15099545_368549362.shtml
}
GCC/Clang生成和链接静态库/动态库
{
1
2
3
4
5
|
//add.cpp int add( int a, int b) { return a + b; } |
1
2
3
4
5
6
7
8
9
10
11
|
//main.cpp #include <iostream> int add( int a, int b); int main( int argc, const char *argv[]) { std::cout << add(1, 2) << std::endl; return 0; } |
笔者的开发平台: Mac OS X 10.9.2 / Apple LLVM version 5.0 (clang-500.2.79) (based on LLVM 3.3svn)
生成静态库
静态库文件的命名规则是:lib****.a(将****替换成自定义名字),将 add.cpp 编译成一个目标文件,然后打包成为一个静态库,具体操作如下:
1.先将 add.cpp 编译成目标文件 add.o
clang++ -c add.cpp
2.然后使用 ar 命令将目标文件 add.o 打包成为静态库文件 libadd.a
ar -r libadd.a add.o
从上面这些操作中可以看出,生成静态库文件其实就是使用 ar 命令将目标文件打包一下,ar 是 archive 的缩写,意思是归档,有关 ar 的部分参数解释如下:
-r :添加或替换指定的文件到归档中;
ar -r libtest.a test.o
-v :显示冗余的信息;
ar -rv libtest.a test.o
-t :按顺序显示归档中的文件名;
ar -t libtest.a
-d :从归档里删除指定的文件;
ar -d libtest.a test.o
从 ar 命令的参数中可以得知,可以将多个目标文件打包到一个静态库中,并可以随时的添加和删除。
链接静态库
生成完静态库了,然后该如何使用这个生成的静态库文件呢?看下面的具体操作:
1.将 main.cpp 模块编译成为目标文件 main.o
clang++ -c main.cpp
2.将目标文件 main.o 和静态库文件 libadd.a 链接成为可执行文件 main
clang++ main.o -L. -ladd -o main
3.运行可执行文件 main
./main
运行结果:
3
链接静态库时,在 clang++ 链接命令后添加了两个参数,下面具体解释一下:
L. :将当前目录添加至编译器库搜索目录中,如果动态库和静态库同时存在,会优先选择动态库;
-ladd :表示查找静态库名是:libadd.a 或 动态库名是:libadd.so 的库文件进行链接,优先选择动态库;
将这两个参数泛化来讲解:
Ldir :将 dir 添加编译器的库查找路径中,编译器默认仅仅搜索 /usr/lib 和 /usr/local/lib 这两个文件夹;
-lname :查找静态库名是:libname.a 或 动态库名是:libname.so 的库文件进行链接,优先选择动态库;
生成动态库
相比静态库,使用动态库生成的可执行文件更小,看下面如何生成一个动态库:
clang++ test.o -shared -fPIC -o libtest.so
执行完上面的操作后,会生成一个动态库文件:libtest.so,动态库文件的命名规则和静态库一样,只是扩展名变成了 .so,下面具体解释一下生成动态库使用的参数的意义:
-shared :表明生成的文件是动态库文件;
-fPIC :表明生成的动态库是位置独立的代码(Position-independent Code),这个参数笔者也解释不太清楚,可以自己 man 一下或者 Google 一下。
-o :指定生成的文件名;
链接动态库
clang++ main.o -L. -ltest -o main
动态库的链接和静态库一样,见上文。
静态库和动态库的区别:
1.使用静态库生成的可执行文件比动态库大;
2.使用静态库生成的可执行文件执行时可以脱离静态库执行,而使用动态库生成的可执行文件在执行时需要动态库文件(所以它比较大)。
3.动态库可以同时被多个程序共享,节省内存和外存。
.
.
思考:
有没有想过静态库和动态库有什么用?其实看看 /usr/lib 和 /usr/local/lib 中的文件就能明白些许。
}