Linux动态库和静态库有什么区别 (linux动态库与静态库的区别)

随着计算机领域的快速发展和应用程序的增多,动态库和静态库的应用也越来越广泛,特别是在Linux系统中,由于其开放的特性和强大的定制性,更是成为了应用程序开发的重要基础。那么,什么是Linux动态库和静态库?它们有什么区别?在软件开发中应该如何合理选择和应用呢?本文就来详细介绍一下,希望能够为读者提供有帮助的知识。

一、什么是Linux动态库和静态库?

1. 动态库

动态库是指在程序运行时才会被加载到内存中执行的库。它通常以.so或者.dll等后缀名为标识,并可被不同的程序共享使用。当程序初始化时,动态库并没有被加载,直到程序运行时程序需要调用动态库中的某个函数时,动态库才会被动态地加载到内存中,从而被程序调用。这种方式可以大大减小程序的运行文件大小,节省系统内存资源,提高代码的重用性,并更加方便程序的更新和维护。

2. 静态库

静态库是指将所有的程序代码和函数库在编译时就打包到了可执行程序中的一种库。它通常以.a、.lib等后缀名为标识,并不可被不同的程序共享使用。静态库虽然比较占用存储空间,但是其执行速度快,能够将程序的依赖关系全部展开,避免程序的一些不必要的依赖问题,使得程序的可靠性更高。

二、Linux动态库和静态库的区别

1. 存储方式

动态库和静态库的存储方式不同。动态库所包含的函数代码并不被直接加载到可执行程序中,而是在程序运行时才被载入到内存中。而静态库则是将所有的库文件在编译时就已经打包到了可执行程序中。

2. 程序的运行特性

动态库和静态库对程序的运行特性也有影响。动态库的优势在于占用内存资源小、可重用性高,及时更新等,但是动态库的载入和卸载会对程序的运行速度造成一定的影响。而静态库的优势在于单一性强,依赖性小,且程序的执行效率高,但是占用内存大,更新维护麻烦。

3. 应用场景

在实际应用场景中,动态库和静态库应用的场合不同。动态库适用于需要运行效率高,占用内存资源小,更新和维护方便等应用场景,亦适用于相同的函数和库被多个程序共享使用。而静态库适用于需要较高的安全性和稳定性等应用场合,且对程序的执行速度要求较高的情况下使用。

三、合理选择和应用

在软件开发中,如何合理地选择和应用动态库和静态库呢?对此,本文向大家介绍几点建议:

1. 根据需求选择

选择动态库还是静态库,应该根据需求来选择,即要看开发的程序所需要的特征是什么,如稳定性、可维护性、执行速度等方面,选择合适的库进行开发。

2. 尽量少用静态库

虽然静态库具有高效性和可靠性高等优点,但是由于静态库的更新、维护和管理都比较麻烦,同时占用内存资源也比较大,因此在实际开发中应该尽量少用静态库。

3. 动态库可用于插件开发

插件开发是动态库的典型应用场景,相应的插件可以动态的加载到程序中,从而实现灵活的扩展和功能扩充。

4. 重复代码要用动态库

当开发中有多个应用程序需要使用到相似甚至相同的代码段时,为了提高程序的重用性和可维护性,建议采用动态库来实现重复代码的共享。

结语:

本文对于Linux动态库和静态库的基本概念、区别和应用进行了比较详细的介绍,相信对读者有所帮助。在实际的软件开发中,根据实际情况选择适当的库进行开发是非常重要的,本文也为大家提供了一些实用的建议和参考。

相关问题拓展阅读:

关于动态库 静态库 区别与使用 路径查找等

一、引言

我晌段稿们通常把一些公用函数制作成函数库,供其它程序使用。

函数库分为静态库和动态库两种。

通常情况下,对函数库的链接是放在编译时期(compile time)完成的。所有相关的对象文件(object file)与牵涉到的函数库(library)被链接合成一个可执行文件(executable file)。程序在运行时,与函数库再无瓜葛,因为所有需要的函数已拷贝到相应目录下下。所以这些函数库被成为静态库(static libaray),通常文件名为“lib.a”的形式。

其实,我们也可以把对一些库函数的链接载入推迟到程序运行的时期(runtime)。这就是动态链接库(dynamic link library)技术。

二、两者区别:

a,静态库的使用需要:

1 包含一个对应的头文件告知编译器lib文件里面的具体内容

2 设置lib文件允许编译器去查找已经编译好的二进制代码

b,动态库的使用:

程序运行时需要加载动态库,对动态库有依赖性,需要手动加入动态库

c,依赖性:

静态链接表示静态性,在编译链接之后, lib库中需要的资源已经在可执行程序中了, 也就是静态存在,没有依赖性了

动态,就是实时性,在运行的时候载入需要的资源,那么必须在运行的时候提供 需要的 动态库,有依赖性, 运行时候没有找到库就不能运行了

d,区别:

简单讲,静态库就是直接将需要的代码连接进可执行程序;动态库就是在需要调用其中的函数时,根据函数映射表找到该函数然后调入堆栈执行。

做成静态库可执行文件本身比较大,但不必附带动态库

做成动态库可执行文件本身比较小,但需要附带动态库

链接静态库,编译的可执行文件比较大,当然可以用strip命令精简一下(如:strip libtest.a),但还是要比链接动态库的可执行文件大。程序运行时间燃灶速度稍微快一点。

静态库是程序运行的时候已经调入内存,不管有没有调用,都会在内存里头。静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库。

其在编译程序时若链接,程序运行时会在系统指定的路径下搜索,然后导入内存,程序一般执行时间稍微长一点,但编译的可执行文件比较小;动态库是程序运行的时候需要调用的时候才装入内存,不需要的时候是不会装入内存的。

动态库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入,因此在程序运行时还需要动态库存在。

三、动态链接库的特点与优势

首先让我们来看一下,把库函数推迟到程序运行时期载入的好处:

1. 可以实现进程之间的资源共享。

什么概念呢?就是说,某个程序的在运行中要调用某个动态链接库函数的时候,操作系统首先会查看所有正在运行的程序,看在内存里宴孝是否已有此库函数的拷贝了。如果有,则让其共享那一个拷贝;只有没有才链接载入。这样的模式虽然会带来一些“动态链接”额外的开销,却大大的节省了系统的内存资源。C的标准库就是动态链接库,也就是说系统中所有运行的程序共享着同一个C标准库的代码段。

2. 将一些程序升级变得简单。用户只需要升级动态链接库,而无需重新编译链接其他原有的代码就可以完成整个程序的升级。Windows 就是一个很好的例子。

3. 甚至可以真正坐到链接载入完全由程序员在程序代码中控制。

程序员在编写程序的时候,可以明确的指明什么时候或者什么情况下,链接载入哪个动态链接库函数。你可以有一个相当大的软件,但每次运行的时候,由于不同的操作需求,只有一小部分程序被载入内存。所有的函数本着“有需求才调入”的原则,于是大大节省了系统资源。比如现在的软件通常都能打开若干种不同类型的文件,这些读写操作通常都用动态链接库来实现。在一次运行当中,一般只有一种类型的文件将会被打开。所以直到程序知道文件的类型以后再载入相应的读写函数,而不是一开始就将所有的读写函数都载入,然后才发觉在整个程序中根本没有用到它们。

静态库:在编译的时候加载生成目标文件,在运行时不用加载库,在运行时对库没有依赖性。

动态库:在目标文件运行时加载,手动加载,且对库有依赖性。

具体在开发中用到哪种库,我觉得还是根据实际的内存大小,ROM大小,运行的速度等综合考虑。

关于c/c++静态库和动态库的区别

静态库

之所以成为【静态库】,是因为在链接阶段,会将汇编生成的目标文件.o与引用到的库一起链接打包到可执行文件中。因此对应的链接方式称为静态链接。

试想一下,静态库与汇编生成的目标文件一起链接为可执行文件,那么静态库必定跟.o文件格式相似。其实一个静态库可以简单看成是一组目标文件(.o/.obj文件)的,即很多目标文件经过压缩打包后形成的一个文件。静态库特点总结:

l 静态库对函数库的链接是放在编译时期完成的。

l 程序在运行时与函数库再无瓜葛,移植方便。

l 浪费空间和资源,因为所有相关的目标文件与牵涉到的函数库被链接合成一个可执行文件。

下面编写一些简单的四则运算C++类,将其编译孙陵敬成静态库给他人用,头文件如下所示:

StaticMath.h头文件

#pragma once

class StaticMath

{

public:

StaticMath(void);

~StaticMath(void);

static double add(double a, double b);//加法

static double sub(double a, double b);//减法

static double mul(double a, double b);//乘法

static double div(double a, double b);//除法

void print();

};

Linux下使用ar工具、Windows下vs使用lib.exe,将目标文件压缩到一起,并且对其进行编号和索引,以便于查找和检索。一般创建静态库的步骤如图所示:

图:创建静态库过程

Linux下创建与使用静态库

Linux静态库命名规则

Linux静态库命名规则慎范,必须是”lib.a”:lib为前缀,中间是静态库名,扩展名为.a。

创建静态库(.a)

通过上面的流程可以知道,Linux创建静态库过程如下:

l 首先,将代码文件编译成目标文件.o(StaticMath.o)

g++ -c StaticMath.cpp

注意带参数-c,否则直接编译为可执行文件

l 然后,通过ar工具将目标文件打包成.a静态库文件

ar -crv libstaticmath.a StaticMath.o

生成静态库libstaticmath.a。

大一点的项目会编写makefile文件(CMake等等工程管理工具)来生成静态库,输入多个命令太麻烦了。

使用静态库

编写使用上面创建的静态库的测试代码:

测试代码:

#include “StaticMath.h”

#include

using namespace std;

int main(int argc, char* argv)

{

double a = 10;

double b = 2;

cout

using namespace std;

int main(int argc, char* argv)

{

double a = 10;

double b = 2;

cout

using namespace std;

int _tmain(int argc, _TCHAR* argv)

{

double a = 10;

double b = 2;

cout ,提供了下面几个接口:

l void * dlopen( const char * pathname, int mode ):函数以指定模式打开指定的动态连接库文件,并返回一个句柄给调用进程。

l void* dlsym(void* handle,const char* symbol):dlsym根据动态链接库操作句柄(pHandle)与符号(symbol),返回符号对应的地址。使用这个函数不但可以获取函数地址,也可以获取变量地址。

l int dlclose (void *handle):dlclose用于关闭指定句柄的动态链接库,只有当此动态链接库的使用计数为0时,才会真正被系统卸载。

l const char *dlerror(void):当动态链接库操作函数执行失败时,dlerror可以返回出错信息,返回值为NULL时表示操作函数执行成功。

在Windows下显式调用动态库

应用程序必须进行函数调用以在运行时显式加载 DLL。为显式链接到 DLL,应用程序必须:

l 调用 LoadLibrary(或相似的函数)以加载 DLL 和获取模块句柄。

l 调用 GetProcAddress,以获取指向应用程序要调用的每个导出函数的函数指针。由于应用程序是通过指针调用 DLL 的函数,编译器不生成外部引用,故无需与导入库链接。

l 使用完 DLL 后调用 FreeLibrary。

显式调用C++动态库注意点

对C++来说,情况稍微复杂。显式加载一个C++动态库的困难一部分是因为C++的name

mangling;另一部分是因为没有提供一个合适的API来装载类,在C++中,您可能要用到库中的一个类,而这需要创建该类的一个实例,这不容易做到。

name mangling可以通过extern “C”解决。C++有个特定的关键字用来声明采用C

binding的函数:extern “C” 。用 extern “C”声明的函数将使用函数名作符号名,就像C函数一样。因此,只有非成员函数才能被声明为extern

“C”,并且不能被重载。尽管限制多多,extern “C”函数还是非常有用,因为它们可以象C函数一样被dlopen动态加载。冠以extern

linux动态库与静态库的区别的介绍就聊到这里吧,感谢你花时间阅读本站内容,更多关于linux动态库与静态库的区别,Linux动态库和静态库有什么区别,关于动态库 静态库 区别与使用 路径查找等,关于c/c++静态库和动态库的区别的信息别忘了在本站进行查找喔。


数据运维技术 » Linux动态库和静态库有什么区别 (linux动态库与静态库的区别)