一.              动态链接库的原理及使用

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 使用与编译的更多相关文章

  1. Linux 上GCC的静态编译和动态编译

    静态编译 常规编译示例: $gcc xxx.c yyy.c zzz.c -o rslt 注明: gcc编译器会对源文件min.c进行预处理, 编译, 以及链接, 最后生成可执行文件 $gcc -c x ...

  2. Linux下librdkafka客户端的编译运行

    Linux下librdkafka客户端的编译运行 librdkafka是一个开源的Kafka客户端C/C++实现,提供了Kafka生产者.消费者接口. 由于项目需要,我要将Kafka生产者接口封装起来 ...

  3. 在linux下用命令行编译 java的eclipse项目

    由于jdk的版本问题导致在windows上编译打包好的jar包放在linux服务器上运行的时候出现一点小异常,所以决定在linux上进行一次项目编译,这有两个选择1.在相同的linux环境下安装lin ...

  4. Linux下通过源码编译安装程序(configure/make/make install的作用,然后在/etc/profile文件里修改PATH环境变量)

    一.程序的组成部分 Linux下程序大都是由以下几部分组成: 二进制文件:也就是可以运行的程序文件 库文件:就是通常我们见到的lib目录下的文件 配置文件:这个不必多说,都知道 帮助文档:通常是我们在 ...

  5. Linux中源码安装编译Vim

    Linux中源码安装编译Vim Linux下学习工作少不了编辑器,Vim能使你的工作效率成倍的提高.在Ubuntu上安装vim使用命令直接安装很简单.但有时还是需要自己手动编译安装.例如: vim中的 ...

  6. Linux下FFmpeg的安装编译过程【转】

    本文转载自:http://www.linuxidc.com/Linux/2013-06/85628.htm 详细说下在Linux下FFmpeg的安装编译过程.参考 Ubuntu 10.04安装编译FF ...

  7. [转]Caffe在Linux下的安装,编译,实验

    Caffe在Linux下的安装,编译,实验  原文地址:http://www.cnblogs.com/evansyang/p/6150118.html 第一部分:Caffe 简介 caffe是有伯克利 ...

  8. 【转载】在Linux中使用VS Code编译调试C++项目

    原文:在Linux中使用VS Code编译调试C++项目 最近项目需求,需要在Linux下开发C++相关项目,经过一番摸索,简单总结了一下如何通过VS Code进行编译调试的一些注意事项. 关于VS ...

  9. Linux环境PostgreSQL源码编译安装

    Linux环境PostgreSQL源码编译安装 Linux版本: Red Hat 6.4 PostgreSQL版本: postgresql-9.3.2.tar.gz 数据存放目录: /var/post ...

随机推荐

  1. What does addScalar do?

    The JavaDoc says: SQLQuery org.hibernate.SQLQuery.addScalar(String columnAlias, Type type) Declare a ...

  2. POJ 1953

    //FINBONACI数列 #include <iostream> #define MAXN 100 using namespace std; int _m[MAXN]; int main ...

  3. jstl 的应用 java

    JSTL :JSP Standard Tag Library,JSP标准标签库 1.导入包 jstl.jar standard.jar 2.页面中添加标识 <%@taglib uri=" ...

  4. C# 任意类型数据转JSON格式

    /// <summary> /// List转成json /// </summary> /// <typeparam name="T">< ...

  5. redis资料汇总

    redis资源比较零散,引用nosqlfan上的文章,方便大家需要时翻阅.大家看完所有的,如果整理出文章的,麻烦知会一下,方便学习. 1.Redis是什么? 十五分钟介绍 Redis数据结构 Redi ...

  6. C++函数默认参数

    C++中允许为函数提供默认参数,又名缺省参数. 使用默认参数时的注意事项: ① 有函数声明(原型)时,默认参数可以放在函数声明或者定义中,但只能放在二者之一 double sqrt(double f ...

  7. Hadoop基础教程之高级编程

    从前面的学习中,我们了解到了MapReduce整个过程需要经过以下几个步骤: 1.输入(input):将输入数据分成一个个split,并将split进一步拆成<key, value>. 2 ...

  8. ios开发--企业帐号发布

    这两天需要发布一个ipa放到网上供其他人安装,需要用到企业级开发者账号. 首先详细说明一下我们的目标,我们需要发布一个ipa放到网上,所有人(包括越狱及非越狱设备)可以直接通过链接下载安装,不需要通过 ...

  9. Java:进制转换

    进制转换是常常需要的一种数据处理,在java中的一些类中封装了具有转换功能的方法,这个不做介绍.其实,进制之间的转化是通过先位异或&,再位移动>>>的方式实现的. 例如,对于 ...

  10. linux 命令案例学习——文件搜索

    两个搜索文件的工具 locate  ——仅仅通过文件名查找文件 find     ——依据文件的各种属性在既定目录(包括子目录)里查找 一个通常与文件搜索命令一起使用.处理搜索结果文件列表的命令 xa ...