本文结合一个简单示例,说明了linux的动态链接库的创建和使用问题,其中容易混淆的是编译链接时的库文件路径和运行期的库文件路径问题。
1、创建自定义动态链接库文件(.so)和头文件
(1) 源文件编写。包括动态库的源文件 fred.c ,头文件fred.h 以及调用程序main.c
库文件实现源文件 fred.c
#include <stdio.h>
void fred()
{
printf("fred is running../n");
}
库文件声明头文件 fred.h
#ifndef __FRED_H__
#define __FRED_H__
#include <stdio.h>
void fred();
#endif
调用程序 main.c
#include "fred.h"
int main()
{
fred();
return 0;
}
(2)生成动态库
命令: gcc -o libhello.so -O2 -fPIC -shared hello.c
其中-o 表示输出文件名为libhello.so,动态链接库名称必须按照libXX.so格式命名;-O2表示使用编译器优化;
库文件、库申明、调用程序都在同一文件夹中
2、使用自定义动态链接库
(1) 编译 gcc -c -o main.c main.o
(2) 链接 gcc -o main main.o -L. -lfred
ldconfig做的这些东西都与运行程序时有关,跟编译时一点关系都没有。编译的时候还是该加-L就得加,不要混淆了
如果硬是要求不用使用-L参数,则可以在LD_LIBRARY_PATH中添加库文件所在的路径,如:
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:./
gcc -o main main.o -lfred
(3) 运行 ./main
通过在/etc/ld.so.conf.d/下添加一个包含自libfred.so绝对路径的文件,并且运行ldconfig命令
如果没有运行ldconfig,则运行main时出现找不到库文件libfred的错误。
(4) 总结:
使用动态链接库的程序要想正确运行,需要注意程序不同时期的路径问题:
a、链接期路径: 通过-L或者LD_LIBRARY_PATH环境变量指定,使得gcc可以找到动态链接库文件
b、运行期路径: 通过在/etc/ld.so.conf.d/下添加一个包含自定义链接库绝对路径的文件,并且运行ldconfig命令,通过ldconfig -p|grep XXX,来查看库文件是否已经被包括了
使得ld可以在程序运行时可动态加载库文件
注意二者必须同时满足时候,调用程序才可以正常运行。
我开始以为在ld.so.conf.d/中添加路径文件并运行ldconfig之后就可以不用-L参数了,结果所以导致程序一直链接不通过,汗。。。。