1、动态库*.so的编译
这里我们用到4个文件,它们分别为:SoDemoTest.h、one.cpp、two.cpp、three.cpp。它们的内容如下:
SoDemoTest.h
#ifndef __SO_DEMO_TEST_HEADER__
#define __SO_DEMO_TEST_HEADER__
#include <iostream>
using namespace std;
void one();
void two();
void three();
#endif
one.cpp
#include "SoDemoTest.h"
void one()
{
cout << "call one() function." << endl;
}
two.cpp
#include "SoDemoTest.h"
void two()
{
cout << "call two() function." << endl;
}
three.cpp
#include "SoDemoTest.h"
void three()
{
cout << "call three() function." << endl;
}
将这几个文件编译成动态库libtest.so。编译命令如下:
$ g++ one.cpp two.cpp three.cpp -fPIC -shared -o libtest.so
2、动态库的连接
在上面的部分,我们已经生成了一个libtest.so
的动态链接库,现在我们用一个程序来调用这个动态链接库。文件名为:main.cpp
main.cpp
#include "SoDemoTest.h"
int main()
{
one();
two();
three();
return 0;
}
将main.cpp
与libtest.so
链接成一个可执行文件main
。命令如下:
$ g++ main.cpp -L. -ltest -o main
测试可执行程序main
是否已经链接动态库libtest.so
,如果列出了libtest.so
,那么就说明正常链接了。可以执行以下命令:
$ ldd main
执行main
可以看看main
是否调用了动态链接库中的函数。
3、编译参数
-shared
该选项指定生成动态连接库(让连接器生成T类型的导出符号表,有时候也生成弱连接W类型的导出符号),不用该标志外部程序无法连接。相当于一个可执行文件
-fPIC
:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。
-L.
:表示要连接的库在当前目录中
-ltest
:编译器查找动态连接库时有隐含的命名规则,即在给出的名字前面加上lib
,后面加上.so
来确定库的名称
LD_LIBRARY_PATH
:这个环境变量指示动态连接器可以装载动态库的路径。
4、注意的问题
调用动态库的时候有几个问题会经常碰到,有时,明明已经将库的头文件所在目录 通过 “-I” include
进来了,库所在文件通过 “-L”
参数引导,并指定了“-l”
的库名,
但通过ldd
命令察看时,就是死活找不到你指定链接的so
文件,这时你要作的就是通过修改 LD_LIBRARY_PATH
或者/etc/ld.so.conf
文件来指定动态库的目录。
LD_LIBRARY_PATH: 动态库的查找路径
设置:
方法一: export LD_LIBRARY_PATH=LD_LIBRARY_PATH:/XXX 但是登出后就失效
方法二: 修改~/.bashrc或~/.bash_profile或系统级别的/etc/profile
1. 在其中添加例如export PATH=/opt/ActiveP/lib:$LD_LIBRARY_PATH
2. source .bashrc (Source命令也称为“点命令”,也就是一个点符号(.)。source命令通常用于重新执行刚修改的初始化文件,使之立即生效,而不必注销并重新登录)
方法三:这个没有修改LD_LIBRARY_PATH但是效果是一样的实现动态库的查找,
1. /etc/ld.so.conf下面加一行/usr/local/mysql/lib
2. 保存过后ldconfig一下(ldconfig 命令的用途,主要是在默认搜寻目录(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所列的目录下,搜索出可共享的动态链接库(格式如前介绍,lib*.so*),进而创建出动态装入程序(ld.so)所需的连接和缓存文件.缓存文件默认为/etc/ld.so.cache,此文件保存已排好序的动态链接库名字列表.)
方法三设置稍微麻烦,好处是比较不受用户的限制。
通常这样做就可以解决库无法链接的问题了。