浅谈GCC/Clang生成和链接静态库/动态库

为了方便下面的讲解,先写两个C++源文件,代码如下:

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 中的文件就能明白些许。

展开阅读全文

没有更多推荐了,返回首页