linux shared lib 使用与编译
一. 动态链接库的原理及使用
Linux提供4个库函数、一个头文件dlfcn.h以及两个共享库(静态库libdl.a和动态库libdl.so)支持动态链接。
Ø dlopen:打开动态共享目标文件并将其映射到内存中,返回其首地址
Ø dlsym:返回锁请求的入口点的指针
Ø dlerror:返回NULL或者指向描述最近错误的字符串
Ø dlclose:关闭动态共享文件
函数dlopen需要在文件系统中查找目标文件并为之创建句柄。有四种方法指定目标文件的位置:
Ø 绝对路径
Ø 在环境变量LD_LIBRARY_PATH指定的目录中
Ø 在/etc/ld.so.cache中指定的库列表中
Ø 在/usr/lib或者/lib中
下面举一个例子。
主程序文件hello.c:
#include <stdio.h> #include <dlfcn.h> void* slib=0; void (*func)(char*); const char* hError; int main(int argc,char* argv[]) { slib=dlopen("./slib.so",RTLD_LAZY); hError=dlerror(); if (hError) { printf("dlopen Error!/n"); return 1; } func=dlsym(slib,"func"); hError=dlerror(); if (hError) { dlclose(slib); printf("dlsym Error!/n"); return 1; } func("How do you do?/n"); dlclose(slib); hError=dlerror(); if (hError) { printf("dlclose Error!/n"); return 1; } return 0; } |
函数dlopen的第二个参数有两种选择:
Ø RTLD_LAZY:推迟解析DLL中的外部引用,直到DLL被执行
Ø RTLD_NOW:在返回之前解析所有的外部引用
以下是DLL文件源码slib.c:
int func(char* msg) { printf("func be Executed!/n"); printf(msg); return 0; } |
是不是很简单?
源代码写好后,在编译和链接时有点复杂。为此,我们编写了一个makefile:
all:hello slib.so hello: gcc -o hello hello.c -ldl slib.so:slib.o gcc -shared -lc -o slib.so slib.o slib.o: gcc -c -fpic slib.c |
生成这个程序需要三步:
Ø 将DLL编译为位置无代码
Ø 创建DLL共享目标文件
Ø 编译主程序并与DLL相链接
编译slib.c时,使用了-fpic或者-fPIC选项,使生成的代码是位置无关的,因为重建共享目标库需要位置无关,并且这类代码支持大的偏移。
创建DLL共享目标文件时使用了-shared选项,该选项产生适合动态链接的共享目标文件slib.so。
生成主程序时,使用-ldl选项,这是链接选项,即主程序中的部分符号为动态链接库中的符号,也就是说,在运行时需要到dll文件中才能够解决引用。
二. 通用类型的动态函数库的建立
Linux操作系统和各种软件包为软件开发人员提供了很多的动态函数库文件。但是一般情况下这些库还不能满足用户的所有需求。开发人员会根据自己的需要编写很多的函数。对于这些函数,如果总是将源文件与调用它们的程序链接起来,虽然也可以,但是,缺点是显然的。下面就将它们加入动态函数库中。
在Linux中,建立动态函数库不需要额外的工具,只需要gcc就可以了。
通过ldd命令可以很方便的察看程序用到了哪些库。
下面通过一个简单的例子说明动态函数库的建立过程。
文件mylib.c是函数库的源程序文件,内容如下:
int myadd(int a1, int a2) { return a1+a2; } |
文件testlib.c是测试程序的源程序文件:
#incoude <stdio.h> extern int myadd(int, int); int main() { printf(“%d/n”,myadd(1, 2)); return 0; } |
下面给出makefile的内容:
all:libmylib.so.1.0 testlib libmylib.so.1.0 : mylib.o ld –m elf_i386 –shared –soname libmylib.so.1 –o libmylib.so.1.0 mylib.o ln –sf libmylib.so.1.0 libmylib.so.1 ln –sf libmylib.so.1 libmylib.so testlib : testlib.c gcc –Wall –O2 –L. –lmylib –o testlib testlib.c mylib.o : mylib.c gcc –c –Wall –O2 –fPIC –o mylib.o mylib.c clean : -rm –f libmylib.so* testlib *.o |
在Linux的shell中输入make命令,动态函数库libmylib.so.1.0和它的测试程序就生成了。运行./testlib试试看。
如果你不走运的话,系统会提示找不到libmylib.so.1动态函数库,因为系统认为没有这样的文件或目录。不要慌。你可能需要使用LD_LIBRARY_PATH环境变量。
[root@localhost home]export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH
再运行一次测试程序吧。
linux shared lib 使用与编译的更多相关文章
- Linux 上GCC的静态编译和动态编译
静态编译 常规编译示例: $gcc xxx.c yyy.c zzz.c -o rslt 注明: gcc编译器会对源文件min.c进行预处理, 编译, 以及链接, 最后生成可执行文件 $gcc -c x ...
- Linux下librdkafka客户端的编译运行
Linux下librdkafka客户端的编译运行 librdkafka是一个开源的Kafka客户端C/C++实现,提供了Kafka生产者.消费者接口. 由于项目需要,我要将Kafka生产者接口封装起来 ...
- 在linux下用命令行编译 java的eclipse项目
由于jdk的版本问题导致在windows上编译打包好的jar包放在linux服务器上运行的时候出现一点小异常,所以决定在linux上进行一次项目编译,这有两个选择1.在相同的linux环境下安装lin ...
- Linux下通过源码编译安装程序(configure/make/make install的作用,然后在/etc/profile文件里修改PATH环境变量)
一.程序的组成部分 Linux下程序大都是由以下几部分组成: 二进制文件:也就是可以运行的程序文件 库文件:就是通常我们见到的lib目录下的文件 配置文件:这个不必多说,都知道 帮助文档:通常是我们在 ...
- Linux中源码安装编译Vim
Linux中源码安装编译Vim Linux下学习工作少不了编辑器,Vim能使你的工作效率成倍的提高.在Ubuntu上安装vim使用命令直接安装很简单.但有时还是需要自己手动编译安装.例如: vim中的 ...
- Linux下FFmpeg的安装编译过程【转】
本文转载自:http://www.linuxidc.com/Linux/2013-06/85628.htm 详细说下在Linux下FFmpeg的安装编译过程.参考 Ubuntu 10.04安装编译FF ...
- [转]Caffe在Linux下的安装,编译,实验
Caffe在Linux下的安装,编译,实验 原文地址:http://www.cnblogs.com/evansyang/p/6150118.html 第一部分:Caffe 简介 caffe是有伯克利 ...
- 【转载】在Linux中使用VS Code编译调试C++项目
原文:在Linux中使用VS Code编译调试C++项目 最近项目需求,需要在Linux下开发C++相关项目,经过一番摸索,简单总结了一下如何通过VS Code进行编译调试的一些注意事项. 关于VS ...
- Linux环境PostgreSQL源码编译安装
Linux环境PostgreSQL源码编译安装 Linux版本: Red Hat 6.4 PostgreSQL版本: postgresql-9.3.2.tar.gz 数据存放目录: /var/post ...
随机推荐
- JavaScript文件应该放在网页的什么位置
JavaScript文件应该放在网页的什么位置 http://www.lihuai.net/qianduan/js/352.html 在<良好的JavaScript编程习惯>系列教程的第三 ...
- 学习android的博客
http://www.cnblogs.com/hll2008/http://svn.apache.org/repos/asf/http://blog.csdn.net/chenzheng_javaht ...
- root 授权
错误:The user specified as a definer ('root'@'%') does not exist 解决: grant all privileges on *.* to ro ...
- 堆排序 Heap Sort
堆排序虽然叫heap sort,但是和内存上的那个heap并没有实际关系.算法上,堆排序一般使用数组的形式来实现,即binary heap. 我们可以将堆排序所使用的堆int[] heap视为一个完全 ...
- oci.dll文件是用来干嘛的? 如果没有安装ORACLE客户端提示oci.dll未加载
oracle数据库开发编程中,没有找到oci.dll,一般是系统的 path 设置有问题, 查找oci.dll, 然后加入到系统路径.oci.dll 可下载解压到系统盘的system32目录下.然后打 ...
- C#计算时间差值
/// <summary> /// 计算时间差值 /// </summary> /// <param name="DateTime1">< ...
- lintcode :最长公共子串
题目 最长公共子串 给出两个字符串,找到最长公共子串,并返回其长度. 样例 给出A=“ABCD”,B=“CBCE”,返回 2 注意 子串的字符应该连续的出现在原字符串中,这与子序列有所不同. 解题 注 ...
- FreePascal经典资料
------------------------------------------------------------------------ 这是每个版本的changelog: http://bu ...
- Largest Rectangle in Histogram-最大长方形
题目描述: 给定n个非负整数height[n],分别代表直方图条的高,每个条的宽设为1,求直方图中面积最大的矩形的面积 题目来源: http://oj.leetcode.com/problems/la ...
- C++:delete和delete[]释放内存的区别
C++告诉我们在回收用 new 分配的单个对象的内存空间的时候用 delete,回收用 new[] 分配的一组对象的内存空间的时候用 delete[]. 关于 new[] 和 delete[], ...