以下实例的环境是amd64 + ubuntu10.10 + g++ 4.4.5测试成功,在其他配置的机器上可能有一点区别。
动态库的使用方式中有两种,第一种是类似于静态库的使用,另一种我称之为真正的动态加载动态库,主要是因为这种方式在程序运行的过程中加载链接库,使用之后在卸载链接库。
先介绍第一种。
在目录/home/owner/test/下创建我们的实验程序:
//dll_fun.c
#include <stdio.h>
void dll_function(const char* szString)
{
printf("%s
", szString);
}
编译生成动态链接库
gcc -c -fPIC dll_fun.c //这里一定要加上-fPIC选项,不然下一步编译失败
gcc -shared -fPIC -o libdllfun.so dll_fun.o //生成动态链接库libdllfun.so
创建调用动态库方法:
//main.c
void dll_function(const char* szString);
int main()
{
dll_function("This is the words of the dll function!!!!!!");
return 0;
}
编译main.c生成可执行文件
gcc -o main main.c -L. -ldllfun //这里提供了刚才生成的dllfun库
如果此时执行./main的话,会出现如下错误:
cannot open shared object file: No such file or directory
这是因为系统未找到动态库libdllfun.so。
Linux动态链接库的默认搜索路径是/lib和/usr/lib,因此动态库被创建后,一般都复制到这两个目录下面,当程序执行时需要某动态库,并且改动态库还没有加载到内存中,则系统会自动到这两个默认的搜索路径中去查找相应的动态库文件,然后加载改文件到内存中,这样程序就可以使用该动态库中的函数以及该动态库中的其他资源了。在linux中,动态库的搜索路径除了默认的搜索路径外,还可以通过其他三种方法来指定,这里只介绍其中的一种:通过环境变量LD_LIBRARY_PATH指定动态库搜索路径。
当通过该环境变量指定多个动态链接库搜索路径时,路径之间用冒号":"分隔。
使用下面命令来配置环境
mkdir /home/owner/test/lib//将这个目录设置为动态库的存放目录
mkdir/home/owner/test/libdllfun.so /home/owner/test/lib/libdllfun.so
export LD_LIBRARY_PATH=/home/owner/test/lib
此时设置这个环境变量之后的所有命令命令中,该环境变量都有效。
./main
可得如下结果:
This is the words of the dll function!!!!!!
第二种加载动态库实例:
//dll_fun.c
#include<stdio.h>
void dll_function(const char* szString)
{
printf("%s
", szString);
}
编译该文件:
gcc -Wall -fPIC -c dll_fun.c
gcc -shared -W1,-soname,libdllfun.so.1 -o libdllfun.so.1.0 *.o
sudo mv libdllfun.so.1.0 /usr/lib
sudo ln -sf /usr/lib/libdllfun.so.1.0 /usr/lib/libdllfun.so
sudo ln -sf /usr/lib/libdllfun.so.1.0 /usr/lib/libdllfun.so.1
参数详解:
-Wall:包含warning信息
-fPIC:编译动态库所必须的,输出不依赖位置的代码
-shared:编译动态库必须选项
-W1:向连接器传递一些参数,在这里传递的参数有“-soname”,"libdllfun.so.1"
-o:动态库的名字,在这个例子里最终生成动态库libdllfun.so.1.0
两个符号链接的含义:
第一个:允许应用代码用-dllfun的语法进行编译