[整理]LINUX动态链接库高级应用.
Linux下Makefile中动态链接库和静态链接库的生成与调用

Linux下Makefile中动态链接库和静态链接库的⽣成与调⽤Linux下Makefile中动态链接库和静态链接库的⽣成与调⽤ 背景:写这篇博客的原因是:最近在搞嵌⼊式,需要交叉编译opencv库⽂件,⾃⼰写Makefile,通过arm-linux-g++编译、链接、⽣成可执⾏⽂件,从⽽实现了移植的过程。
平台是Toradex的Apalis TK1,三千多元,买回来我就后悔了,全是英⽂资料,还各种Bug,迟迟⽆法上⼿。
早知如此,还不如直接买Nvidia的Jetson TK1呢。
书归正传,今天写⼀下Makefile⽂件中,动态链接库和静态链接库的⽣成与调⽤。
⼀、概念 动态链接库:是⼀种不可执⾏的⼆进制程序⽂件,它允许程序共享执⾏特殊任务所必需的代码和其他资源。
Windows平台上动态链接库的后缀名是”.dll”,Linux平台上的后缀名是“.so”。
Linux上动态库⼀般是libxxx.so;相对于静态函数库,动态函数库在编译的时候并没有被编译进⽬标代码中,你的程序执⾏到相关函数时才调⽤该函数库⾥的相应函数,因此动态函数库所产⽣的可执⾏⽂件⽐较⼩。
由于函数库没有被整合进你的程序,⽽是程序运⾏时动态的申请并调⽤,所以程序的运⾏环境中必须提供相应的库。
动态函数库的改变并不影响你的程序,所以动态函数库的升级⽐较⽅便。
静态链接库:这类库的名字⼀般是libxxx.a;利⽤静态函数库编译成的⽂件⽐较⼤,因为整个函数库的所有数据都会被整合进⽬标代码中,他的优点就显⽽易见了,即编译后的执⾏程序不需要外部的函数库⽀持,因为所有使⽤的函数都已经被编译进去了。
当然这也会成为他的缺点,因为如果静态函数库改变了,那么你的程序必须重新编译。
Makefile:利⽤IDE开发调试的⼈员可能对Makefile不⼤理解,其实Makefile就是完成了IDE中的编译、链接、⽣成等⼯作,并遵循shell脚本中变量定义与调⽤的规则。
⼆、编写Makefile实现编译与链接1、准备⽂件 我们写⼀个简单的⼯程吧,此⼯程包含3个⽂件,分别是main.cpp,func.cpp和func.h。
如何在Linux下调试动态链接库

如何在Linux下调试动态链接库⼤家都知道在 Linux 可以⽤ gdb 来调试应⽤程序,当然前提是⽤ gcc 编译程序时要加上-g 参数。
我这篇⽂章⾥将讨论⼀下⽤ gdb 来调试动态链接库的问题。
⾸先,假设我们准备这样的⼀个动态链接库:QUOTE:库名称是: ggg动态链接库⽂件名是: libggg.so头⽂件是: get.h提供这样两个函数调⽤接⼝:int get ();int set (int a);要⽣成这样⼀个动态链接库,我们⾸先编写这样⼀个头⽂件:[Copy to clipboard]CODE:/************关于本⽂档*********************************************filename: get.h*purpose: ⼀个动态链接库头⽂件⽰例*tided by: zhoulifa() 周⽴发 ()Linux 爱好者 Linux 知识传播者 SOHO 族开发者最擅长 C 语⾔*date time: 2006-11-15 21:11:54*Note: 任何⼈可以任意复制代码并运⽤这些⽂档,当然包括你的商业⽤途* 但请遵循 GPL*Hope:希望越来越多的⼈贡献⾃⼰的⼒量,为科学技术发展出⼒* 科技站在巨⼈的肩膀上进步更快!感谢有开源前辈的贡献!*感谢提供原始代码,我在他的基础上整理了此⽂*********************************************************************/int get ();int set (int a);然后准备这样⼀个⽣成动态链接库的源⽂件:[Copy to clipboard]CODE:/************关于本⽂档*********************************************filename: get.cpp*purpose: ⼀个动态链接库源⽂件⽰例*tided by: zhoulifa() 周⽴发 ()Linux 爱好者 Linux 知识传播者 SOHO 族开发者最擅长 C 语⾔*date time:2006-11-15 21:11:54*Note: 任何⼈可以任意复制代码并运⽤这些⽂档,当然包括你的商业⽤途* 但请遵循 GPL*Hope:希望越来越多的⼈贡献⾃⼰的⼒量,为科学技术发展出⼒* 科技站在巨⼈的肩膀上进步更快!感谢有开源前辈的贡献!*感谢提供原始代码,我在他的基础上整理了此⽂*********************************************************************/#include <stdio.h>#include "get.h"static int x=0;int get (){printf ("get x=%d\n", x);return x;}int set (int a){printf ("set a=%d\n", a);x = a;return x;}然后我们⽤ GNU 的 C/C++ 编译器来⽣成动态链接库,编译命令如下:QUOTE:g++ get.cpp -shared -g -DDEBUG -olibggg.so这样我们就准备好了动态链接库了,下⾯我们编写⼀个应⽤程序来调⽤此动态链接库,源代码如下:[Copy to clipboard]CODE:/************关于本⽂档*********************************************filename: pk.cpp*purpose: ⼀个调⽤动态链接库的⽰例*tided by: zhoulifa() 周⽴发 ()Linux 爱好者 Linux 知识传播者 SOHO 族开发者最擅长 C 语⾔*date time:2006-11-15 21:11:54*Note: 任何⼈可以任意复制代码并运⽤这些⽂档,当然包括你的商业⽤途* 但请遵循 GPL*Hope:希望越来越多的⼈贡献⾃⼰的⼒量,为科学技术发展出⼒* 科技站在巨⼈的肩膀上进步更快!感谢有开源前辈的贡献!*感谢提供原始代码,我在他的基础上整理了此⽂*********************************************************************/#include <stdio.h>#include "get.h"int main (int argc, char** argv){int a = 100;int b = get ();int c = set (a);int d = get ();printf ("a=%d,b=%d,c=%d,d=%d\n",a,b,c,d);return 0;}编译此程序⽤下列命令,如果已经把上⾯⽣成的 libggg.so 放到了库⽂件搜索路径指定的⽂件⽬录,⽐如 /lib 或 /usr/lib 之类的,就⽤下⾯这条命令:QUOTE:g++ pk.cpp -o app -Wall -g -lggg否则就⽤下⾯这条命令:g++ pk.cpp -o app -Wall -g -lggg -L`pwd`下⾯我们就开始调试上⾯命令⽣成的 app 程序吧。
探秘Linux链接库:静态与动态的交响诗

探秘Linux链接库:静态与动态的交响诗《探秘Linux链接库:静态与动态的交响诗》在Linux系统的宏伟乐章中,链接库扮演着至关重要的角色,它们是程序构建的基石,是系统运行的润滑剂。
静态链接库与动态链接库,这两种看似对立的技术,实则相辅相成,共同谱写了一曲美妙的交响诗。
一、静态链接库:稳固的基石静态链接库,以其稳定性和可靠性著称。
它们在编译时期便与应用程序紧密结合,确保了程序的一致性和性能。
如同坚实的地基,静态链接库为应用程序提供了稳固的支撑,使其在运行时无需额外的依赖,独立而高效。
二、动态链接库:灵活的翅膀与静态链接库的稳重相比,动态链接库则展现出了更多的灵活性和活力。
它们允许程序在运行时动态加载所需的库,实现了资源共享和内存的节省。
动态链接库如同一双翅膀,赋予了程序飞翔的能力,使其能够灵活地适应不同的运行环境。
三、链接的艺术:静态与动态的和谐共舞在Linux的世界里,静态与动态链接库并非孤立存在,而是相互补充,共同构建了强大的系统。
静态链接库确保了基础功能的稳定实现,而动态链接库则为程序带来了扩展和更新的可能性。
它们之间的和谐共舞,展现了Linux系统设计的艺术之美。
四、环境变量的指挥棒:LD_LIBRARY_PATH在动态链接库的演奏中,环境变量LD_LIBRARY_PATH 扮演着指挥棒的角色。
它指引着程序寻找动态链接库的方向,确保了演奏的顺利进行。
正确的设置LD_LIBRARY_PATH,就如同为交响乐团设定了正确的节拍,让每一个音符都能准确无误地奏响。
五、ldd的洞察力:揭示依赖的真相ldd命令如同一位洞察力极强的侦探,它能够揭示程序背后的依赖关系,让隐藏在程序背后的动态链接库无处遁形。
通过ldd的侦查,开发者可以清晰地了解程序的构成,为程序的调试和优化提供了有力的线索。
六、结语:链接库的未来展望随着技术的发展,链接库的角色将变得更加多样化。
我们期待在未来的Linux系统中,链接库能够提供更加高效、灵活、安全的服务。
Linux下静态链接库和动态链接库的使用

Linux下静态链接库和动态链接库的使用库有动态与静态两种,动态通常用.so为后缀,静态用.a为后缀。
例如:libtest.so libtest.a。
为了在同一系统中使用不同版本的库,可以在库文件名后加上版本号为后缀,但由于程序连接默认以.so为文件后缀名。
所以为了使用这些库,通常使用建立符号连接的方式。
如:ln -s libtest.so.1.0 hello.so.1ln -s libtest.so.1 hello.so 下面对比一下两者:静态链接库:当要使用时,连接器会找出程序所需的函数,然后将它们拷贝到执行文件,由于这种拷贝是完整的,所以一旦连接成功,静态程序库也就不再需要了。
动态库而言:某个程序在运行中要调用某个动态链接库函数的时候,操作系统首先会查看所有正在运行的程序,看在内存里是否已有此库函数的拷贝了。
如果有,则让其共享那一个拷贝;只有没有才链接载入。
在程序运行的时候,被调用的动态链接库函数被安置在内存的某个地方,所有调用它的程序将指向这个代码段。
因此,这些代码必须使用相对地址,而不是绝对地址。
在编译的时候,我们需要告诉编译器,这些对象文件是用来做动态链接库的,所以要用地址不无关代码(Position Independent Code (PIC))。
注意:linux下进行连接的缺省操作是首先连接动态库,也就是说,如果同时存在静态和动态库,不特别指定的话,将与动态库相连接。
下面就通过实际的例子来向大家演示一下,该怎样编译和使用静态和动态链接库,这里有一个头文件:lib_test.h,一个.c文件:lib_test.c:1.编写库文件lib_test.c2.编写一个头文件用于声明我们使用的函数lib_test.h3.用gcc编绎该文件,可以使用任何合法的编绎参数#include<stdio.h> #ifndef _LIBTEST_H gcc -c lib_test.c -o lib_test.ovoid a() #define _LIBTEST_H{ void a();printf("zyx\n"); #endif}下面就用上面的文件生成和使用静态和动态链接库:(一)静态链接库1)用gcc编绎该文件,可以使用任何合法的编绎参数gcc -c lib_test.c -o lib_test.o2) $ar crv libtest.a lib_test.o //生成静态库生成libtest.a3) 在某些系统中还要为静态库生成一个内容表$ranlib libtest.a4) 使用静态链接库$nm libtest.a //nm工具可以打印出库中的涉及到的所有符号,库既可以是静态的也可以是动态的。
Linux命令高级技巧使用ln命令创建文件和的链接

Linux命令高级技巧使用ln命令创建文件和的链接Linux命令高级技巧:使用ln命令创建文件和链接Linux是一种开源的操作系统,其强大的命令行工具为用户提供了丰富的功能。
其中,ln命令是一个非常有用的工具,可以用于创建文件和链接。
本文将介绍如何使用ln命令来创建文件和链接,并提供一些高级技巧。
1. 创建硬链接硬链接是文件或目录的副本,它们与原始文件或目录共享相同的inode和数据块。
这意味着,无论是原始文件还是硬链接,对它们的更改都会反映在彼此之间。
要创建硬链接,可以使用以下命令:```ln 源文件链接文件```例如,我们有一个名为file1.txt的文件,要在同一目录下创建一个名为link1.txt的硬链接,可以运行以下命令:```ln file1.txt link1.txt```现在,如果我们对file1.txt进行更改,link1.txt也会反映这些更改。
2. 创建软链接软链接也被称为符号链接,它是一个指向文件或目录的快捷方式。
与硬链接不同,软链接保存的是文件或目录的路径,而不是实际的数据块。
如果原始文件或目录被删除,软链接将失效。
要创建软链接,可以使用以下命令:```ln -s 源文件链接文件```例如,我们有一个名为file2.txt的文件,要在同一目录下创建一个名为link2.txt的软链接,可以运行以下命令:```ln -s file2.txt link2.txt```现在,通过打开link2.txt,我们可以访问file2.txt的内容。
3. 创建目录链接除了文件链接,ln命令还可以用于创建目录链接。
目录链接允许我们在不同的位置上访问相同的目录。
要创建目录链接,可以使用以下命令:```ln -s 源目录链接目录```例如,我们有一个名为dir1的目录,要在同一级目录下创建一个名为link_dir1的目录链接,可以运行以下命令:```ln -s dir1 link_dir1```现在,我们可以通过访问link_dir1来访问dir1目录下的文件。
LINUX下动态链接库的创建和使用浅谈

Linux 下动态链接库的创建和使用浅谈李兴飞2011-9-6一直以来,我对Linux下的动态链接库的理解比较模糊,由于要在工程中需要使用动态链接库,所以这几天对这块进行了扫盲。
我在学习Linux环境下的动态使用时,是带着下面三个问题进行的:(1)什么是动态链接库,作用是什么?(2)使用c编写动态库,c程序如何调用动态库?(3)使用c++编写动态库,c程序如何调用动态库?(4)是否可以将c和c++分别别写的代码一个整合到动态库中?一、什么是动态链接库?作用是什么?动态链接库是一个可以被其它应用程序共享的程序模块,其中封装了一些可以被共享的例程和资源。
动态链接库的扩展名是.so。
作用:使用动态链接库可以有效的利用空间和资源,使应用程序可以共享相同功能的代码模块。
二、使用c编写动态库,c程序如何调用动态库?下面采用例子介绍:(1)首先编写动态链接库的对外暴露API的头文件test_c.h,我们这里定义的API是test_c_api() 函数,代码如下:>>>>>>>>>>>>>>>>>>>>>>test_c.h>>>>>>>>>>>>>>>>>>>>>>#ifndef __TEST_C_H#define __TEST_C_H#include <stdio.h>void test_c_api();#endif动态链接库的实现部分,下边是一个打印hello word的源代码:>>>>>>>>>>>>>>>>>>>>>>test_c.c>>>>>>>>>>>>>>>>>>>>>>#include "test_c.h"void test_c_api(){printf("test_c hello word!\n");}(2)编译生成动态链接库:gcc –c –fPIC test_c.c 产生test_c.o 目标文件gcc –shared –o libtest.so test_c.o 生成动态链接库libtest.so注意:-fPIC 选项的意思是产生于本地无关的代码-shared选项生产动态链接库也可以直接使用gcc –fPIC –shared –o libtest.so test_c.c 生成动态链接库。
LinuxC.动态库和静态库以及库链接
LinuxC.动态库和静态库以及库链接================================ 动态库和静态库 ===================================== 库文件是什么?printf -->有谁见过这个函数的内部实现?头文件用来做什么?举例:你们公司做摄像头的,你卖摄像头会不会把源码卖了?不卖源码,别人要用所以把源码打包成为库文件camera.c —>编译成为库文件,比如libcamera.socamera.h —>存放相应C文件函数接口说明只给看的到的函数文件接口说明和看不到内容的库文件一、静态库1)用来做什么?2)静态库格式:libxxx.a3)将 add.c sub.c 编译为静态库文件,操作步骤如下:①先把.c文件生成.o文件(机器码)gcc -c add.c -o add.ogcc -c sub.c -o sub.o②再把.o文件合成静态库ar cr libstatic.a add.o sub.o使用静态库:原来的操作:gcc main.c add.c sub.c -o target 现在的操作:gcc main.c -o target -lstatic -L. --》头文件和库文件都在当前路径-lstatic —>链接库 l —>link库的格式:libxxx.a链接库的格式:-lxxx-L. —>告诉编译器库文件的路径以下方式常用:gcc main.c -o target -lstatic -L /home/gec/lib -I/home/gec/include/home/gec/lib —>存放 libstatic.a 文件/home/gec/include —>存放 add.h sub.h-L —》指定库文件的路径-I —》指定头文件的路径还有一种不建议的操作:将相应的库文件头文件放入系统默认的路径 : /usr/lib /libgec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$ sudo cp ~/lib/libstatic.a /usr/lib[sudo] password for gec:gec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$ sudo cp ~/include/*.h /usr/include/gec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$gec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$ gcc main.c -o target -lstatic —>当前编译不需要指定路径还有一种办法:将库文件和头文件所在路径添加为默认路径(环境变量)LD_LIBRARY_PATHLIBRARY_PATH —》设置的是编译时库文件的路径C_INCLUDE_PATH —》设置编译时头文件的路径查看环境变量echo $LD_LIBRARY_PATHecho $LIBRARY_PATHecho $C_INCLUDE_PATH设置环境变量export LIBRARY_PATH=path:$LIBRARY_PATH —>设置库文件路径gec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$ echo $ LIBRARY_PATHgec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$ export LIBRARY_PATH=/home/gec/lib:LIBRARYPATHgec@ubuntu:/mnt/ hgfs/share/day03/code/staticLib echo LIBRARYPATH/home/gec/ lib:gec@ubuntu:/mnt/hgfs/share/day03/code/staticLib export C_INCLUDE_PATH=pathgec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$ echo $C_INCLUDE_PATHgec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$ export C_INCLUDE_PATH=/home/gec/includegec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$ echo CINCLUDEPATH/home/gec/includegec@ubuntu:/mnt/hgfs/shar e/day03/code/staticLib gcc main.c -o target -lstatic —>前面已经设置过头文件和库文件路径,所以不需要指定路径gec@ubuntu:/mnt/hgfs/share/day03/code/staticLib$注意:使用静态库一旦编译得到可执行程序之后就不需要静态库了,可执行文件可以独立运行缺点就是可执行文件太大二、动态库 1)格式:libxxx.so 2)将 add.c sub.c 编译为动态库文件,操作步骤如下:gcc -c add.c -o add.o -fPIC gcc -c sub.c -o sub.o -fPICgcc -fPIC -shared add.o sub.o -o libdyname.so编译时的第一种办法:gcc main.c -o target -ldyname -I/home/gec/include-L/home/gec/lib —》指定动态库文件和头文件的路径gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$ gcc main.c -o target -ldyname -I/home/gec/include -L/home/gec/lib gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$ ./target ./targ et: error while loading shared libraries: libdyname.so: cannot open shared object file: No such file or directory gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$编译时的第二种办法:—》设置相应的路径为环境变量exportLIBRARY_PATH=/home/gec/lib:LIBRARYPATH−−−>设置库文件路径exportCINCLUDEPATH=/home/gec/includegec@ubuntu:/mnt/h gfs/share/day03/code/dyLib gcc main.c -o target-ldyname —》编译时不需要指定头文件和库文件的路径gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$ ./target ./targ et: error while loading shared libraries: libdyname.so: cannot open shared object file: No such file or directory —》运行时无法链接到动态库的路径gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$ echo $LD_LIBRARY_PATHgec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$ export LD_LIBRARY_PATH=/home/gec/lib:LDLIBRARYPATH−−−》将动态库的路径加入环境变量gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib echo LDLIBRA RYPATH/home/gec/lib:gec@ubuntu:/mnt/hgfs/share/day03/cod e/dyLib gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$ ./tar get1 sum = 11 sub = 22 gec@ubuntu:/mnt/hgfs/share/day03/code/dyLib$注意:动态库编译出来的可执行文件在运行时依然需要链接动态库三、总结1)静态库如何编译得到一个静态库编译时的第一种办法:直接指定库文件和头文件路径编译时的第二种办法:设置头文件和库文件所在路径为环境变量使用 LIBRARY_PATH —》设置静态库所在的路径为环境变量使用 C_INCLUDE_PATH —》设置头文件所在的路径为环境变量要想长期有效将相应的命令设置到启动脚本文件,比如:~/.bashrc /etc/profile2)动态库如何编译得到一个动态库编译时的第一种办法:直接指定库文件和头文件路径编译时的第二种办法:设置头文件和库文件所在路径为环境变量使用 LIBRARY_PATH —》设置静态库所在的路径为环境变量使用 C_INCLUDE_PATH —》设置头文件所在的路径为环境变量要想长期有效将相应的命令设置到启动脚本文件,比如:~/.bashrc /etc/profile注意:运行时还是需要链接动态库,所以需要使用LD_LIBRARY_PATH 设置链接路径3)静态库与动态库的对比静态库:生成可执行文件后,优点是不再依赖于静态库可独立运行,缺点是库都添加到执行文件里了体积大。
linux下编译动态链接库(.so文件)的步骤和示例代码
在Linux下,我们可以使用GCC编译器来编译动态链接库(.so文件)。
以下是一个简单的示例,演示了如何创建一个动态链接库。
假设我们有一个简单的C程序,它定义了一个函数add,该函数接受两个整数并返回它们的和。
c// add.c#include<stdio.h>int add(int a, int b) {return a + b;}接下来,我们需要编写一个头文件add.h,以供其他程序使用。
c// add.h#ifndef ADD_H#define ADD_Hint add(int a, int b);#endif// ADD_H现在,我们编写一个动态链接库程序libadd.c。
该程序包含了上面定义的头文件和函数。
c// libadd.c#include"add.h"int add(int a, int b) {return a + b;}编译动态链接库的步骤如下:1.使用gcc编译器将C源文件编译为共享对象文件。
使用-shared选项指定输出为共享对象文件。
例如:2.shellgcc -shared -o libadd.so add.c libadd.c这将生成一个名为libadd.so的共享对象文件。
2. 将生成的共享对象文件复制到适当的位置。
例如,您可以将文件复制到系统的库目录中:shellsudo cp libadd.so /usr/local/lib/这将将共享对象文件复制到/usr/local/lib/目录下,这样其他程序就可以找到并链接它。
请注意,可能需要使用超级用户权限来执行此操作。
linux shell下的静态链接库和动态链接库实例(脚本)
开发一个库函数,要求能对字符串的进行大写、小写转换。
要求把库函数,编译成静态和动态链接库各一个,取名libtu.a,libtu.so。
编写一个测试程序,测试库函数。
注:本次所创建的文件均在同一目录下,程序的编译通过脚本文件new@ubuntu:~/xi4$ vim daxiao.c //创建大小转换函数new@ubuntu:~/xi4$ vim main.c //创建主函数new@ubuntu:~/xi4$ main.h 创建头文件[1]静态链接库new@ubuntu:~/xi4$ vim g1.sh //创建脚本文件,用来建立静态库并生成可执行程序,其后缀为“.sh”//(标准的库名均以lib开头,静态库后缀为”.a”;-I用来指明头文件main.h路径,“.”指当前的;-L指明加载库的路径-l用来链接标准的表达方式,不加“lib”)new@ubuntu:~/xi4$ su //进入超级管理员修改权限Passwd //输入隐式密码root@ubuntu:/home/new/xi4# chmod +rwx g1.sh//脚本文件执行需赋予执行权限root@ubuntu:/home/new/xi4# ./g1.sh //执行脚本文件root@ubuntu:/home/new/xi4# ./m1 //执行可执行程序[2]动态链接库root@ubuntu:/home/new/xi4# vim g2.sh//建立第一个脚本文件//标准动态链接库以lib开头,以“.so”为后缀注意参数“-shared”“-fpic”注意空格root@ubuntu:/home/new/xi4# chmod +rwx g2.sh//脚本文件执行需赋予执行权限root@ubuntu:/home/new/xi4# ./run.sh//执行脚本文件此处不能直接执行生成的.m2可执行程序需建立另一个脚本文件root@ubuntu:/home/new/xi4# vim run.sh//建立第二个脚本文件//此处采用隐式链接动态库的方法,通过设置环境变量LD_LIBRARY_PATH,把库所在路径加入这个变量中root@ubuntu:/home/new/xi4# chmod +rwx run.sh//脚本文件执行需赋予执行权限root@ubuntu:/home/new/xi4# ./run.sh//执行脚本文件。
linux下静态链接库和动态链接库编译和使用
Linux 下编译链接动静态库2010年03月29日星期一 17:11Linux 版本是 Red Hat 9 ,内核版本是 2.4.18输入 which gcc 查看 gcc 的位置在 /usr/bin/gccgcc -v 查看 gcc 编译前的配置信息--prefix=/usr 说明了安装目录没有 --with-headers 说明默认的 include 就在安装目录下所以 gcc 默认的 include 目录是 /usr/include ,要包含另一个目录,可以用 -I dir 选项包含该目录,想要更方便的可以在 /etc/profile 中添加一个环境变量 C_INCLUDE_PATHC_INCLUDE_PATH="your include path"export C_INCLUDE_PATHgcc 默认的 lib 目录很多,一般是 /lib 和 /usr/lib可以输入 gcc -print-search-dirs 查看同样可以在编译时通过 -L dir 来添加,也可以在 /etc/profile 中添加LD_LIBRARY_PATH="your ldlib path"export LD_LIBRARY_PATH还有就是可以/etc/ld.so.conf中添加目录,这对于安装别的库很方便当然修改了库文件后需要运行一下ldconfig自己制作交叉编译工具太复杂了,直接下一个arm-linux-gcc-3.4.1.tar.bz2 tar jxvf arm-linux-gcc-3.4.1.tar.bz2 -C / 解压缩到根目录下其实由于压缩包带的目录是 usr/local/arm/3.4.1所以实际还是在 /usr/local/arm/3.4.1 目录下在bin中可以看到各个工具 arm-linux-gcc ...输入 ./arm-linux-gcc -v 可以看到配置信息有 --with-headers=/usr/local/arm/3.4.1/arm-linux/include 说明了默认的include目录输入 ./arm-linux-gcc -print-search-dirs 查看搜索的 lib 目录,主要的库文件还是在/usr/local/arm/3.4.1/arm-linux/lib目录下.arm-linux-gcc 3.4.1 可以用来编译2.6的内核而编译bootloader还是用原来的2.95.2版的 arm-linux-gcc程序的预处理、编译、链接都可以由gcc完成,gcc会自动调用cpp来做预处理,ld来进行链接。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
------------- ------------- 在《LINUX下动态链接库的创建与应用》 一文中,我介绍了LINUX动态链接库的基本知识.其要点是:用户根据实际情况需要,利用dlopen,dlsym,dlclose等动态链接库操作函 数,装入指定的动态链接库中指定的函数,然后加以执行.程序中使用很少的动态函数时,这样的做法尚可.如果程序需要调用大量的动态函数,那么采用这样的编 程手段将是非常繁复的,所以我们必须使用一种更为聪明的办法,以减少代码量,提高工作效率.这就是现在我要举例介绍的《LINUX动态链接库高级应用》.
注:本文举的例子类似上篇文章,只是文件的内容已做相应修改,裁减了不少.示例程序ady.c和两个动态函数的源程序getdate.c与gettime.c仅修改了头文件的名字,其内容不再列出.本文使用头文件为adatetime.h.
要想高效地应用LINUX动态链接库(尤其是用户自己编写的),需要做以下工作: 一、编写合格的动态链接库头文件 C语言的头文件,可供一个或多个程序引用,里面一般定义程序所需的常量,自定义类型及函数原型说明等.其中的函数原型说明,则供编译器检查语法,用于排除引用参数时类型不一致的错误.只有编写合格的动态链接库头文件,程序员才能正确使用动态链接库内的函数.
动态链接库头文件要采用C语言标准格式,其中的动态函数原型定义,不必象上文介绍的那样用(*动态函数名)的描述形式.请看下面的例子:(每行开始的数字为所在行行号,为笔者添加,供注解使用)
1 /* adatetime.h : 纵横软件制作中心雨亦奇(zhsoft@371.net)编写, 2002-03-06. */
2 3 #ifndef __DATETIME_H 4 5 #define __DATETIME_H 6 7 /* 日期结构 */ 8 typedef struct ------------- ------------- 9 { 10 int year; 11 int mon; 12 int day; 13 }DATETYPE; 14 15 /* 时间结构 */ 16 typedef struct 17 { 18 char hour; 19 char min; 20 char sec; 21 }TIMETYPE; 22 23 int getdate(DATETYPE *d); /* 取当前日期 */ 24 int gettime(TIMETYPE *t); /* 取当前时间 */ 25 26 #endif 27
注:与上文的datetime.h文件比较,从该头文件第23,24行可以看到,动态函数getdate,gettime的原型定义改变了,不再使用(*getdate),(*gettime)的格式了(这种格式使用较为罗嗦).
二、正确编译与命名动态链接库 为了让GCC编译器生成动态链接库,编译时须加选项-shared.(这点须牢记) LINUX系统中,为了让动态链接库能被系统中其它程序共享,其名字应符合“lib*.so*”这种格式.如果某个动态链接库不符合此格式,则LINUX的动态链接库自动装入程序(ld.so)将搜索不到此链接库,其它程序也无法共享之.
格 式中,第一个*通常表示为简写的库名,第二个*通常表示为该库的版本号.如:------------- ------------- 在我的系统中,基本C动态链接库的名字为libc.so.6,线程 pthread动态链接库的名字为libpthread.so.0等等.本文例子所生成的动态链接库的名字为libmy.so,虽没有版本号,但也符合所 要求的格式.
生成该动态链接库的维护文件makefile-lib内容如下: 1 # makefile : 纵横软件制作中心雨亦奇编写, 2002-03-07. 2 3 all : libmy.so 4 5 SRC = getdate.c gettime.c 6 7 TGT = $(SRC:.c=.o) 8 9 $(SRC) : adatetime.h 10 @touch $@ 11 12 %.o : %.c 13 cc -c $? 14 15 # 动态链接库(libmy.so)生成 16 libmy.so : $(TGT) 17 cc -s -shared -o $@ $(TGT) 18
运行命令: $ make -f makefile-lib $
即生成libmy.so库. 注: 维护文件中,第17行用-shared选项以生成动态链接库,用-s选项以去掉目标文件中的符号表,从而减小文件长度. ------------- ------------- 三、共享动态链接库 3.1 动态链接库配置文件 为 了让动态链接库为系统所使用,需要维护动态链接库的配置文件--/etc/ld.so.conf.此文件内,存放着可被LINUX共享的动态链接库所在 目录的名字(系统目录/lib,/usr/lib除外),各个目录名间以空白字符(空格,换行等)或冒号或逗号分隔.一般的LINUX发行版中,此文件均 含一个共享目录/usr/X11R6/lib,为X window窗口系统的动态链接库所在的目录.
下面看看我的系统中此文件的内容如何: # cat /etc/ld.so.conf /usr/X11R6/lib /usr/zzz/lib #
由上可以看出,该动态库配置文件中,增加了一个/usr/zzz/lib目录.这是我自己新建的共享库目录,下面存放我新开发的可供系统共享的动态链接库.
3.2 动态链接库管理命令 为了让动态链接库为系统所共享,还需运行动态链接库的管理命令--ldconfig.此执行程序存放在/sbin目录下.
ldconfig 命令的用途,主要是在默认搜寻目录(/lib和/usr/lib)以及动态库配置文件/etc/ld.so.conf内所列的目录下,搜 索出可共享的动态链接库(格式如前介绍,lib*.so*),进而创建出动态装入程序(ld.so)所需的连接和缓存文件.缓存文件默认为 /etc/ld.so.cache,此文件保存已排好序的动态链接库名字列表.
ldconfig通常在系统启动时运行,而当用户安装了一个新的动态链接库时,就需要手工运行这个命令.
ldconfig命令行用法如下: ------------- ------------- ldconfig [-v|--verbose] [-n] [-N] [-X] [-f CONF] [-C CACHE] [-r ROOT] [-l] [-p|--print-cache] [-c FORMAT] [--format=FORMAT] [-V] [-?|--help|--usage] path...
ldconfig可用的选项说明如下: (1) -v或--verbose : 用此选项时,ldconfig将显示正在扫描的目录及搜索到的动态链接库,还有它所创建的连接的名字.
(2) -n : 用此选项时,ldconfig仅扫描命令行指定的目录,不扫描默认目录(/lib,/usr/lib),也不扫描配置文件/etc/ld.so.conf所列的目录.
(3) -N : 此选项指示ldconfig不重建缓存文件(/etc/ld.so.cache).若未用-X选项,ldconfig照常更新文件的连接.
(4) -X : 此选项指示ldconfig不更新文件的连接.若未用-N选项,则缓存文件正常更新.
(5) -f CONF : 此选项指定动态链接库的配置文件为CONF,系统默认为/etc/ld.so.conf.
(6) -C CACHE : 此选项指定生成的缓存文件为CACHE,系统默认的是/etc/ld.so.cache,此文件存放已排好序的可共享的动态链接库的列表.
(7) -r ROOT : 此选项改变应用程序的根目录为ROOT(是调用chroot函数实现的).选择此项时,系统默认的配置文件/etc/ld.so.conf,实际对应的为 ROOT/etc/ld.so.conf.如用-r /usr/zzz时,打开配置文件/etc/ld.so.conf时,实际打开的是/usr/zzz/etc/ld.so.conf文件.用此选项,可以 大大增加动态链接库管理的灵活性.
(8) -l : 通常情况下,ldconfig搜索动态链接库时将自动建立动态链接库的连接.选择此项时,将进入专家模式,需要手工设置连接.一般用户不用此项.
(9) -p或--print-cache : 此选项指示ldconfig打印出当前缓存文件所保存的所有共享库的名字. ------------- ------------- (10) -c FORMAT 或 --format=FORMAT : 此选项用于指定缓存文件所使用的格式,共有三种:old(老格式),new(新格式)和compat(兼容格式,此为默认格式).
(11) -V : 此选项打印出ldconfig的版本信息,而后退出. (12) -? 或 --help 或 --usage : 这三个选项作用相同,都是让ldconfig打印出其帮助信息,而后退出.
举三个例子: 例1: # ldconfig -p 793 libs found in cache `/etc/ld.so.cache' libzvt.so.2 (libc6) => /usr/lib/libzvt.so.2 libzvt.so (libc6) => /usr/lib/libzvt.so libz.so.1.1.3 (libc6) => /usr/lib/libz.so.1.1.3 libz.so.1 (libc6) => /lib/libz.so.1 ...... #
注: 有时候用户想知道系统中有哪些动态链接库,或者想知道系统中有没有某个动态链接库,这时,可用-p选项让ldconfig输出缓存文件中的动态链接库列 表,从而查询得到.例子中,ldconfig命令的输出结果第1行表明在缓存文件/etc/ld.so.cache中找到793个共享库,第2行开始便是 一系列共享库的名字及其全名(绝对路径).因为实际输出结果太多,为节省篇幅,以......表示省略的部分.
例2: # ldconfig -v /lib: