静态库和动态库的区别
1. 静态函数库
这类库的名字一般是libxxx.a;利用静态函数库编译成的文件比较大,因为整个 函数库的所有数据都会被整合进目标代码中,他的优点就显而易见了,即编译后的执行程序不需要外部的函数库支持,因为所有使用的函数都已经被编译进去了。当然这也会成为他的缺点,因为如果静态函数库改变了,那么你的程序必须重新编译。
2. 动态函数库
这类库的名字一般是libxxx.so;相对于静态函数库,动态函数库在编译的时候 并没有被编译进目标代码中,你的程序执行到相关函数时才调用该函数库里的相应函数,因此动态函数库所产生的可执行文件比较小。由于函数库没有被整合进你的程序,而是程序运行时动态的申请并调用,所以程序的运行环境中必须提供相应的库。动态函数库的改变并不影响你的程序,所以动态函数库的升级比较方便。
linux系统有几个重要的目录存放相应的函数库,如/lib /usr/lib。
动态库制作
实例说明:
假设当前目录有一下文件:
/////////////main.c//////#include<math2.h>头文件是我自己的
1 #include<stdio.h>
2 #include<math2.h>
3 int main(){
4 int i=11,j=22;
5 printf("__%d__ ",myadd(i,j));
6 return 0;
7 }
////////math2.h///////////////////
1 #ifndef _MATH2_H_
2 #define _MATH2_H_
3 int myadd(int ,int );
4 #endif
/////////////////myadd.c//////////
1 int myadd(int i,int j){
2 return i+j;
3 }
对于头文件这里用两种情况:
情况一:将其放到/usr/include/
情况二:在编译时通过-I来指定需要用到的头文件在那个目录
动态库制作使用:
现在我要将myadd.c制作成动态库,使得main.c能够使用该动态库
第一步:gcc -c myadd.c -o myadd.o -fpic
//用到fpic参数的原因:myadd.o需要来制作库文件,库文件不同于一般
的二进制文件,他需要一些特殊的格式和特殊的信息,因此用-fpic来生成这些特
殊格式和信息
第二步:gcc -shared -WL,-soname,libmath.so.1 -o libmath-1.0.0.so
myadd.o
//-shared表示要生成动态库;-wl,-sonname,libmyadd.so.1将会导致字
符串libmyadd.so.1作为文件名称被记录在最终生成的库文件libmath-1.0.0.so中
。
ok至此生成了该动态库,但是此时还不能使用,需要下面在做一些工作:
有两种情况使用该动态库:
情况一:将其放到系统库中
第一步:cp libmath-1.0.0.so /lib/
注意此时用到的库名是math;
第二步:但是此时通过gcc main.c -lmath还是不能编译成功,这是因为,编译器
到系统库中找到库,是找libmath.so,而不是直接找libmath-1.0.0.so,因而需要
在/lib/下建立libmath-1.0.0.so的软连接:
ln -s /lib/libmath-1.0.0.so libmath.so
第三步:在gcc编译后的文件还是不能执行,这是因为,我们在制作动态库的时候
指定了链接的库名是libmath.so.1(但是我们真正的库是libmath-1.0.0.so),因
而我们还需要创建一个关于libmath-1.0.0.so的软连接:
ln -s /lib/libmath-1.0.0.so libmath.so.1
上面的工作做完之后就可以完全的将制作的动态库使用在我们的应用程序,即编
译和运行都可以通过
情况二:可以将头文件以及生成的库,还有关于该库的两个软连接放到当前(或
者你指定的目录下);
第一步:将情况二说的文件生成并一一放到当前目录下
第二步:此时编译需要用到:
gcc main.c -L. -I.
//对于-L是用于指定在编译时需要用到库搜索路径 //对于-I是用于指定
在编译时头文件的存放目录
第三步:在编译成功后的文件还是不能直接运行,需要添加一个环境变量在指定
在链接库时的搜索路劲(注意不能写错了变量名):
export LD_LIBRARY_PATH=/hyg/linux高级编程/文件IO/jingtai
完了之后在自己设定的目录下就可以运行自己生成的动态库文件。
静态库的制作:
情况一:放到系统库中
第一步: gcc -c myadd.c -o myadd.o
第二步:ar rc libmyadd.a myadd.o
//第二步是制作静态库
第三步:cp libmyadd.a /lib/
第四步:gcc mian.c -lmyadd -I. -static
情况二:可以放到指定目录,在编译时需要指定路径,方法同动态库的情况二
此下面部分转于:http://www.cnblogs.com/feisky/archive/2010/03/09/1681996.html
动态库的显式调用
显式调用动态库需要四个函数的支持, 函数 dlopen 打开动态库, 函数 dlsym 获取动态库中对象基址, 函数 dlerror 获取显式动态库操作中的错误信息, 函数 doclose 关闭动态库.
[bill@billstone make_lib]$ cat main.c
#include <dlfcn.h>
int main()
{
void *pHandle;
void (*pFunc)(); // 指向函数的指针
int *p;
pHandle = dlopen("./d1.so", RTLD_NOW); // 打开动态库
if(!pHandle){
printf("Can't find d1.so ");
exit(1);
}
pFunc = (void (*)())dlsym(pHandle, "print"); // 获取库函数 print 的地址
if(pFunc)
pFunc();
else
printf("Can't find function print ");
p = (int *)dlsym(pHandle, "p"); // 获取库变量 p 的地址
if(p)
printf("p = %d ", *p);
else
printf("Can't find int p ");
dlclose(pHandle); // 关闭动态库
return 0;
}
[bill@billstone make_lib]$ gcc -o tds main.c –ld1 –L.
此时还不能立即./tds,因为在动态函数库使用时,会查找/usr/lib、/lib目录下的动态函数库,而此时我们生成的库不在里边。 这个时候有好几种方法可以让他成功运行: 最直接最简单的方法就是把libstr_out.so拉到/usr/lib或/lib中去。 还有一种方法 export LD_LIBRARY_PATH=$(pwd) 另外还可以在/etc/ld.so.conf文件里加入我们生成的库的目录,然后/sbin/ldconfig。 /etc/ld.so.conf是非常重要的一个目录,里面存放的是链接器和加载器搜索共享库时要检查的目录,默认是从/usr/lib /lib中读取的,所以想要顺利运行,我们也可以把我们库的目录加入到这个文件中并执行/sbin/ldconfig 。另外还有个文件需要了解/etc/ld.so.cache,里面保存了常用的动态函数库,且会先把他们加载到内存中,因为内存的访问速度远远大于硬盘的访问速度,这样可以提高软件加载动态函数库的速度了。
库依赖的查看
使用ldd命令来查看执行文件依赖于哪些库。
该命令用于判断某个可执行的 binary 档案含有什么动态函式库。
[root@test root]# ldd [-vdr] [filename]
参数说明:
--version 打印ldd的版本号
-v --verbose 打印所有信息,例如包括符号的版本信息
-d --data-relocs 执行符号重部署,并报告缺少的目标对象(只对ELF格式适用)
-r --function-relocs 对目标对象和函数执行重新部署,并报告缺少的目标对象和函数(只对ELF格式适用)
--help 用法信息。
如果命令行中给定的库名字包含'/',这个程序的libc5版本将使用它作为库名字;否则它将在标准位置搜索库。运行一个当前目录下的共享库,加前缀"./"。