制作动态链接库
main.c
#include <stdio.h> #include "vendor.h" int add_ret(int a , int b) { return a + b ; } int main(void) { int sum = add(3,4,add_ret); printf("sum:%d\n",sum); return 0 ; }
vendor.h
#ifndef __VENDOR_H #define __VENDOR_H int add(int a , int b , int (*add_value)()); #endif
vendor.c
#include "vendor.h" int add(int a , int b , int (*add_value)()) { return (*add_value)(a,b); }
创建动态链接库命令
[root@hello]#gcc -shared -fPIC vendor.c - o libvendor.so [root@hello]#cp libvendor.so /usr/lib [root@hello]#gcc main.c -L . -lvendor -o main [root@hello]#./main
解释:
生成动态库:
gcc -shared -fPIC dvendor.c -o libvendor.so
-shared : 生成动态库;
-fPIC : 生成与位置无关代码;
-o :指定生成的目标文件;
使用动态库:
gcc main.c -L . –lvendor -o main
-L : 指定库的路径(编译时); 不指定就使用默认路径(/usr/lib/lib)
-lvendor : 指定需要动态链接的库是谁;
代码运行时需要加载动态库:
./main 加载动态库 (默认加载路径:/usr/lib /lib ./ ...)
./main
中间会出现以下错误error while loading shared libraries:xxx.so cannnot open shared
解决方案:
原因一般有两个:
一个是操作系统里确实没有包含该共享库(lib*.so.*文件)或者共享库版本不对, 遇到这种情况那就去网上下载并安装上即可.
二是已经安装了该共享库, 但执行需要调用该共享库的程序的时候, 程序按照默认共享库路径找不到该共享库文件.
所以安装共享库后要注意共享库路径设置问题, 如下:
1) 如果共享库文件安装到了/lib或/usr/lib目录下, 那么需执行一下ldconfig命令
ldconfig命令的用途, 主要是在默认搜寻目录(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所列的目录下, 搜索出可共享的动态链接库(格式如lib*.so*), 进而创建出动态装入程序(ld.so)所需的连接和缓存文件. 缓存文件默认为/etc/ld.so.cache, 此文件保存已排好序的动态链接库名字列表.
2) 如果共享库文件安装到了/usr/local/lib(很多开源的共享库都会安装到该目录下)或其它"非/lib或/usr/lib"目录下, 那么在执行ldconfig命令前, 还要把新共享库目录加入到共享库配置文件/etc/ld.so.conf中, 如下:
# cat /etc/ld.so.conf
include ld.so.conf.d/*.conf
# echo "/usr/local/lib" >> /etc/ld.so.conf
# ldconfig
3) 如果共享库文件安装到了其它"非/lib或/usr/lib" 目录下, 但是又不想在/etc/ld.so.conf中加路径(或者是没有权限加路径). 那可以export一个全局变量LD_LIBRARY_PATH, 然后运行程序的时候就会去这个目录中找共享库.
LD_LIBRARY_PATH的意思是告诉loader在哪些目录中可以找到共享库. 可以设置多个搜索目录, 这些目录之间用冒号分隔开. 比如安装了一个mysql到/usr/local/mysql目录下, 其中有一大堆库文件在/usr/local/mysql/lib下面, 则可以在.bashrc或.bash_profile或shell里加入以下语句即可:
export LD_LIBRARY_PATH=/usr/local/mysql/lib:$LD_LIBRARY_PATH
一般来讲这只是一种临时的解决方案, 在没有权限或临时需要的时候使用.