1.静态库
先来编写简单的add.c add.h || sum.c sum.h
add.h
1 #ifndef __ADD_H__
2 #define __ADD_H__
3
4 int add(int a, int b);
5
6 #endif // __ADD_H__
add.c
1.#include "add.h"
2
3 int add(int a, int b)
4 {
5 return a + b;
6 }
~
sum.h
1 #ifndef __SUB_H__
2 #define __SUB_H__
3
4 int sub(int a, int b);
5
6 #endif // __SUB_H__
7
8
sum.c
1 #include "sub.h"
2
3 int sub(int a, int b)
4 {
return a - b;
6 }
1.生成静态库
ar -rc lib静态库名.a 一堆.o文件
2.使用静态库
编译选项
-L:指定库路径
-l:指定库名
t:列出静态库中的文件
v:详细信息
当测试目标文件生成后,静态库删掉,程序照样可以运行
2.动态库
fPTC : 产生位置无关码
shared:表示生成共享库格式
库名规则:libXXX.so
2.动态库的使用
编译选项:
-L:链接库所在的路径
-l:链接动态库,只要苦命即可(去掉lib以及版本号)
将上述的代码进行动态库的生成和进行链接
此时还不能立即a./out,因为在动态函数库使用时,会查找/usr/lib /lib目录下的动态函数库,而此时我们生成的库不在里边。
这个时候有好几种方法可以让他成功运行:
>>1.最直接最简单的方法就是把libstr_out.so拉到/usr/lib 或/lib64中去。
>>2. export LD_LIBRARY_PATH=$(pwd)
>>3.还可以在/etc/ld.so.conf文件里加入我们生成的库的目录,然后/sbin/ldconfig,
ldconfig是刷新动态库路径的缓存
/etc/ld.so.conf是非常重要的一个目录,里面存放的是链接器和加载器搜索共享库时要检查的目录,默认是从/usr/lib /lib中读取的,所以想要顺利运行,我们也可以把我们库的目录加入到这个文件中并执行/sbin/ldconfig
3.为什么使用库
>>静态库解决的问题:解决给客户一堆.o文件的问题,以库的形式提供给客户,在程序加载时可以直接使用库
>> 动态库:解决同样的代码在物理内存上存储多分的问题
>>Linux中有两类函数库,分别是静态库和动态库。
静态函数库:
这类库的名字一般是libxxx.a;利用静态函数库编译成的文件比较大,因为整个函数库的所有数据都会被整合进目标代码中,他的优点就显而易见了, 即编译后的执行程序不需要外部的函数库支持,因为所有使用的函数都已经被编译进去了。当然这也会成为他的缺点,因为如果静态函数库改变了,那么你的程序必 须重新编译。
动态函数库:
这类库的名字一般是libxxx.so;相对于静态函数库,动态函数库在编译的时候并没有被编译进目标代码中,你的程序执行到相关函数时才调用该函数 库里的相应函数,因此动态函数库所产生的可执行文件比较小。由于函数库没有被整合进你的程序,而是程序运行时动态的申请并调用,所以程序的运行环境中必须 提供相应的库。动态函数库的改变并不影响你的程序,所以动态函数库的升级比较方便。
4.动态库函数
动态库的运行时加载:
这几个函数也是以动态库的形式提供给我们,连接的时候使用
#include <dlfcn.h>
//打开动态库文件
void *dlopen(const char *filename, //动态库文件名
int flag);//RTLD_LAZY
//返回值:代表刚刚打开的动态库句柄
//char *dlerror(void);
void *disym(void *handle,//dlopen的返回值
const char *symbol);//要使用的函数名
//返回值;函数的起始地址
//关闭
int dlclose(void *handle);