静态库
1) 直接调用
#gcc main.c libmath.a
2) 或通过LIBRARY_PATH环境变量指定库路径
#export LIBRARY_PATH=$LIBRARY_PATH:.
#gcc main.c -lmath (环境法)
3) 通过gcc的-L选项指定库路径
#unset LIBRARY_PATH
#gcc main.c -lmath -L. (参数法)
一般化的方法:gcc .c/.o -l<库名> -L<库路径>
add.c
#include "add.h"
int add(int a,int b)
{
return a+b;
}
add.h
#ifndef _ADD_H__
#define _ADD_H_
int add(int ,int);
#endif //_ADD_H__
plus.c
#include "plus.h"
int plus(int a,int b)
{
return a*b;
}
plus.h
#ifndef _PLUS_H__
#define _PLUS_H__
int plus(int ,int);
#endif //_PLUS_H__
main.c
#include "add.h"
#include "plus.h"
#include <stdio.h>
int main()
{
int a,b;
a = 10,b = 20;
printf("%d\n",add(a,b));
printf("%d\n",plus(a,b));
return 0;
}
共享库
创建共享库
1) 编辑源程序:.c/.h
2) 编译成目标文件:gcc -c -fpic xxx.c -> xxx.o
3) 链接成共享库文件:gcc -shared xxx.o … -o libxxx.so
# gcc -c -fpic calc.c
# gcc -c -fpic show.c
# gcc -shared calc.o show.o -o libmath.so
//或一次完成编译和链接
# gcc -shared -fpic calc.c show.c -o libmath.so
PIC (Position Independent Code):位置无关代码。可执行程序加载它们时,可将其映射到其地址空间的任何位置。
-fPIC : 大模式,生成代码比较大,运行速度比较慢,所有平台都支持。
-fpic : 小模式,生成代码比较小,运行速度比较快,仅部分平台支持。
注意:提供共享库的同时也需要提供头文件。
2. 调用共享库
1) 直接调用
# gcc main.c libmath.so
2) 通过LIBRARY_PATH环境变量指定库路径
# export LIBRARY_PATH=$LIBRARY_PATH:.
# gcc main.c -lmath (环境法)
3) 通过gcc的-L选项指定库路径
# unset LIBRARY_PATH
# gcc main.c -lmath -L. (参数法)
一般化的方法:gcc .c/.o -l<库名> -L<库路径>
3. 运行
运行时需要保证LD_LIBRARY_PATH,环境变量中包含共享库所在的路径。
在可执行程序的链接阶段,并不将所调用函数的二进制代码复制到可执行程序中。
而只是将该函数在共享库中的地址嵌入到可执行程序中,因此运行时需要依赖共享库。
gcc缺省链接共享库,可通过-static选项强制链接静态库。
八、动态加载共享库
#include <dlfcn.h>
1. 加载共享库
void* dlopen (const char* filename, int flag);
返回值:
成功返回共享库句柄,失败返回NULL。
参数:
filename:共享库路径
若只给文件名,则根据LD_LIBRARY_PATH环境变量搜索。
flag取值:加载方式
RTLD_LAZY - 延迟加载,使用共享库中的符号 (如调用函数)时才加载。
RTLD_NOW - 立即加载。
2.获取函数地址
void* dlsym (void* handle,const char* symbol );
返回值:
成功返回函数地址,失败返回NULL。
参数:
hanedle:共享库句柄
symbol:函数名
3.卸载共享库
int dlclose (void* handle);
成功返回0,失败返回非零。
4. 获取错误信息
.char* dlerror (void);
有错误发生则返回错误信息字符串指针,否则返回NULL