.so Linux下动态库和静态库(.a) 的区别

linux下有两种库:动态库和静态库(共享库)
二者的不同点在于代码被载入的时刻不同 。
静态库的代码在编译过程中已经被载入可执行程序,因此体积比较大 。
动态库(共享库)的代码在可执行程序运行时才载入内存,在编译过程中仅简单的引用,因此代码体积比较小 。
不同的应用程序如果调用相同的库,那么在内存中只需要有一份该动态库(共享库)的实例 。
静态库和动态库的最大区别,静态情况下,把库直接加载到程序中,而动态库链接的时候,它只是保留接口,将动态库与程序代码独立,这样就可以提高代码的可复用度,和降低程序的耦合度 。
静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库 。
动态库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入,因此在程序运行时还需要动态库存在
 
一 静态库
这类库的名字一般是libxxx.a;利用静态函数库编译成的文件比较大,因为整个 函数库的所有数据都会被整合进目标代码中,他的优点就显而易见了,即编译后的执行程序不需要外部的函数库支持,因为所有使用的函数都已经被编译进去了 。当然这也会成为他的缺点,因为如果静态函数库改变了,那么你的程序必须重新编译 。
静态库的代码在编译时链接到应用程序中,因此编译时库文件必须存在,并且需要通过“-L”参数传递路径给编译器,应用程序在开始执行时,库函数代码将随程序一起调入进程内存段直到进程结束,其执行过程不需要原静态库存在 。
在UNIX中,使用ar命令创建或者操作静态库
ar archivefile objfile
archivefile:archivefile是静态库的名称
objfile:objfile是已.o为扩展名的中间目标文件名,可以多个并列
参数 意义
-r 将objfile文件插入静态库尾或者替换静态库中同名文件
-x 从静态库文件中抽取文件objfile
-t 打印静态库的成员文件列表
-d 从静态库中删除文件objfile
-s 重置静态库文件索引
-v 创建文件冗余信息
-c 创建静态库文件
example:
/****************** hello.h **************/    void hello(void);  /****************** hello.cpp **************/    #include<IOStream>  #include"hello.h"  using namespace std;  void hello(void)  {            cout <<"Hello "<<endl;  }  /****************** main.cpp **************/    #include"hello.h"    int main(int argc,char *argv[])  {          hello();          return 0;  }  1.编译成静态库
无论静态库,还是动态库,都是由.o文件创建的 。因此,我们必须将源程序hello.c通过gcc先编译成.o文件 。
hc@linux-v07j:~/weiming/tt> g++ -o hello.o -c hello.cpp
hc@linux-v07j:~/weiming/tt> ar cqs libHello.a hello.o
hc@linux-v07j:~/weiming/tt> ls
hello.cpp hello.h hello.o libHello.a main.cpp
2.链接
hc@linux-v07j:~/weiming/tt> g++ main.cpp libHello.a -o Out1 (g++ -o aOut main.cpp ./libHello.a 或者 g++ -o aOut main.cpp -L./ -lHello)
注意:如果hello() 里面还使用了其他的库函数比如pthread_create,则最后生成Out1 时还需 -lpthread,但ar 时可以不用,只需要在 include 的头文件中找到函数符号声明即可,但最终生成可执行文件时需要找到所有的符号定义 。
hc@linux-v07j:~/weiming/tt> ls
hello.cpp hello.h hello.o libHello.a main.cpp Out1
 
hc@linux-v07j:~/weiming/tt> ldd Out1
linux-gate.so.1 => (0xffffe000)
libstdc++.so.6 => /usr/lib/libstdc++.so.6 (0xb7e36000)
libm.so.6 => /lib/libm.so.6 (0xb7e11000)
libgcc_s.so.1 => /lib/libgcc_s.so.1 (0xb7e06000)
libc.so.6 => /lib/libc.so.6 (0xb7ce3000)
/lib/ld-linux.so.2 (0xb7f1b000)
 
二: 动态库
这类库的名字一般是libxxx.so;相对于静态函数库,动态函数库在编译的时候 并没有被编译进目标代码中,你的程序执行到相关函数时才调用该函数库里的相应函数,因此动态函数库所产生的可执行文件比较小 。由于函数库没有被整合进你的程序,而是程序运行时动态的申请并调用,所以程序的运行环境中必须提供相应的库 。动态函数库的改变并不影响你的程序,所以动态函数库的升级比较方便


推荐阅读