Linux 动态库生成,Linux动态库生成的实例详解

动态库可以节省磁盘和内存的空间,使程序的升级更加容易,增加CPU缓存的命中率等,下面小编通过实例给大家讲解下Linux动态库的生成,一起来了解一下吧。

5cacf4e3b30cda68417d314ff19aef55.png

什么是动态库?

动态库又称动态链接库英文为DLL,是DynamicLinkLibrary的缩写形式,DLL是一个包含可由多个程序同时使用的代码和数据的库,DLL不是可执行文件。动态链接提供了一种方法,使进程可以调用不属于其可执行代码的函数。函数的可执行代码位于一个DLL中,该DLL包含一个或多个已被编译、链接并与使用它们的进程分开存储的函数。DLL还有助于共享数据和资源。多个应用程序可同时访问内存中单个DLL副本的内容。DLL是一个包含可由多个程序同时使用的代码和数据的库。

一、简述

Linux下动态库文件的扩展名为“.so”(SharedObject)。按照约定,所有动态库文件名的形式是libname.so(可能在名字中加入版本号)。这样,线程函数库被称作libthread.so。静态库的文件名形式是libname.a。共享archive的文件名形式是libname.sa。共享archive只是一种过渡形式,帮助人们从静态库转变到动态库。

小编综合自己学习体会以及网络上较好的内容,以简单的例子介绍动态库文件的生成和链接方法。

操作系统:Ubuntu14.0.4

GCC版本:4.1.3

二、库文件及测试文件代码

库文件及测试文件所在的目录:/home/test/program/

1、库文件名:myLibSrc.c

代码如下:

/

filename:myLibSrc.c

/

#include《stdio.h》

#include“myLibInclude.h”

int

myLibSrcFun(){

printf(“ThereismyLibSrcFun()\n”);

return0;

}

2、测试文件:main.c

代码如下:

/

filename:main.c

/

#include《stdio.h》

#include“myLibInclude.h”

intmain(intargc,charargv){

printf(“Mainfunction!\n”);

//调用加载的动态库里的函数

myLibSrcFun();

return0;

}

《strong》3、头文件:myLibInclude.h《/strong》

/

filename:myLibInclude.h

/

intmyLibSrcFun();//声明一下函数

三、动态库的编译方法

编译库文件myLibSrc.c命令如下:

代码如下:

$gccmyLibSrc.c-shared-olibmyLib.so

或是:$gccmyLibSrc.c-fpic-shared-g-DDEBUG-olibmyLib.so

如果编译成功,会在目录/home/test/program/下生成动态库文件:libmyLib.so

这里有两点需要补充说明:

A、对于Linux操作,一般都推荐在普通用户模式下,如果需要超级用户的权限,则可以通过sudo或是suroot,输入root用户密码切换。鉴于个人学习使用时,同时很多操作又需都需要使用root用户,因此就直接在root用户下进行编译。

B、编译生成动态库时的参数含义

-fpic:使输出的对象模块可重定位地址方式生成的。

-shared:指定把对应的源文件生成对应的动态链接库文件。

四、动态库的测试方法

编译测试文件:main.c

代码如下:

$gcc–oappmain.c/home/test/program/libmyLib.so

编译成功后运行。/app:

代码如下:

Mainfunction!

ThereismyLibSrcFun()

需要注意的是:

1、上面编译的命令$gcc–oappmain.c/home/test/program/libmyLib.so

的最后一个参数是指定特定所连接库文件的绝对路径。本例中的库文件的绝对路径即为/home/test/program/libmyLib.so

当然,如果想从系统的库文件路径(通常系统函数库都位于/usr/lib目录下)链接动态库的话,可以先将生成的库文件拷贝至/usr/lib下,然后在链接:

代码如下:

$cplibmyLib.so/usr/lib/libmyLib.so

$gcc–oappmain.c-lmyLib

这里,对于链接的方法做一下简单的解释:对于$gcc–oappmain.c-lmyLib中的最后一个参数–lmyLib,可见传递给C的编译器的命令行参数并未提到函数库的完整路径,甚至没有提到在函数库目录中该文件的完整名字!实际上,编译器被告知根据选项-lmyLib链接到相应的函数库(/usr/lib下),函数库的名字是libmyLib.so,也就是说,“lib”部分和文件的扩展名都被省略了,但在前面加了一个‘l’。

上面就是Linux下动态库生成的实例讲解介绍了,通过这些实例你能够详细了解Linux的动态库是如何生成的,希望对初学者有所帮助。

文中是linux下 C++动态库 实现接口提供类导出的一个例子 注意其中使用函数返回基类指针的用法,因为Linux的动态链接库不能像MFC中那样直接导出类 一、介绍 如何使用dlopen API动态地加载C++函数和类,是Unix C++程序员经常碰到的问题。 事实上,情况偶尔有些复杂,需要一些解释。这正是写这篇mini HOWTO的缘由。 理解这篇文档的前提是对C/C++语言中dlopen API有基本的了解。 这篇HOWTO的维护链接是: http://www.isotton.com/howtos/C++-dlopen-mini-HOWTO/ 二、问题所在 有时你想在运行时加载一个库(并使用其中的函数),这在你为你的程序写一些插件或模块架构的时候经常发生。 在C语言中,加载一个库轻而易举(调用dlopen、dlsym和dlclose就够了),但对C++来说,情况稍微复杂。 动态加载一个C++库的困难一部分是因为C++的name mangling (译者注:也有人把它翻译为“名字毁坏”,我觉得还是不翻译好), 另一部分是因为dlopen API是用C语言实现的,因而没有提供一个合适的方式来装载类。 在解释如何装载C++库之前,最好再详细了解一下name mangling。 我推荐您了解一下它,即使您对它不感兴趣。因为这有助于您理解问题是如何产生的,如何才能解决它们。 1. Name Mangling 在每个C++程序(或库、目标文件)中, 所有非静态(non-static)函数在二进制文件中都是以“符号(symbol)”形式出现的。 这些符号都是唯一的字符串,从而把各个函数在程序、库、目标文件中区分开来。 在C中,符号名正是函数名:strcpy函数的符号名就是“strcpy”,等等。 这可能是因为两个非静态函数的名字一定各不相同的缘故。 而C++允许重载(不同的函数有相同的名字但不同的参数), 并且有很多C所没有的特性──比如类、成员函数、异常说明──几乎不可能直接用函数名作符号名。 为了解决这个问题,C++采用了所谓的name mangling。它把函数名和一些信息(如参数数量和大小)杂糅在一起, 改造成奇形怪状,只有编译器才懂的符号名。 例如,被mangle后的foo可能看起来像foo@4%6^,或者,符号名里头甚至不包括“foo”。 其中一个问题是,C++标准(目前是[ISO14882])并没有定义名字必须如何被mangle, 所以每个编译器都按自己的方式来进行name mangling。 有些编译器甚至在不同版本间更换mangling算法(尤其是g++ 2.x和3.x)。 即使您搞清楚了您的编译器到底怎么进行mangling的,从而可以用dlsym调用函数了, 但可能仅仅限于您手头的这个编译器而已,而无法在下一版编译器下工作。 三、类 使用dlopen API的另一个问题是,它只支持加载函数。 但在C++中,您可能要用到库中的一个类,而这需要创建该类的一个实例,这不容易做到。 四、解决方案 1. extern "C" C++有个特定的关键字用来声明采用C binding的函数: extern "C" 。 用 extern "C"声明的函数将使用函数名作符号名,就像C函数一样。 因此,只有非成员函数才能被声明为extern "C",并且不能被重载。 尽管限制多多,extern "C"函数还是非常有用,因为它们可以象C函数一样被dlopen动态加载。 冠以extern "C"限定符后,并不意味着函数中无法使用C++代码了, 相反,它仍然是一个完全的C++函数,可以使用任何C++特性和各种类型的参数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值