您的位置 首页 模拟

Linux静态库和动态库的编写和运用

库从本质上来说是一种可执行代码的二进制格式,可以被载入内存中执行。库分静态库和动态库两种。1 静态库和动态库的区别 1.1. 静态函数库 (1)静态

库从实质上来说是一种可履行代码的二进制格局,可以被载入内存中履行。库分静态库动态库两种。

1 静态库动态库的差异 1.1. 静态函数库 (1)静态函数库的姓名一般是lib[name].a (2)运用静态函数库编译成的文件比较大,由于整个函数库的一切数据都会被整合进方针代码中,它的长处是编译后的履行程序不需求外部的函数库支撑,由于一切运用的函数都现已被编译进去了。这给它带来的缺陷为假如静态函数库改动了,那么你的程序有必要从头编译。 1.2. 动态函数库 (1)动态函数库的姓名一般是lib[name].so (2)相对于静态函数库,动态函数库所产生的可履行文件比较小,动态函数库在编译的时分 并没有被编译进方针代码中,程序履行到相关函数时才调用该函数库里的相应函数,因而动态函数库所产生的可履行文件比较小。由于函数库没有被整合进你的 程序,而是程序运转时动态的请求并调用,所以程序的运转环境中有必要供给相应的库。动态函数库的改动并不影响你的程序,所以动态函数库的晋级比较便利。 (3)linux体系有几个重要的目录寄存相应的函数库,如/lib /usr/lib,/usr/local/lib。 2 编写及运用静态库

静态库的制造所要用到的指令:gcc和ar 指令。

2.1 编写静态库的过程为: 2.1.1 创立修改源代码文件: pr1.c 和 pr2.c

pr1.c:

void print1()

{

printf(“This is the first lib src!\n”);

}

pr2.c

void print2()

{

printf(“This is the second src lib!\n”);

}

2.1.2 编译.c 文件

gcc -O -c pr1.c pr2.c

2.1.3 链接静态库 在编译程序中正确找到库文件,静态库有必要依照 lib[name].a 的规矩命名,如下例中[name]=pr.

(1)ar -rsv libpr.a pr1.o pr2.o

a – pr1.o

a – pr2.o

(2) ar t ibpr.a

pr1.o

pr2.o

2.1.4 调用库函数代码 main.c

main.c

int main()

{

print1();

print2();

return 0;

}

2.1.5 编译链接选项

-L 及-l 参数放在后边.其间-L 加载库文件途径,-l 指明库文件姓名.

gcc -o main main.c -L./ -lpr

2.1.6 履行方针程序

3 编写动态库 3.1 规划库代码

pr1.c : int p = 2;

void print() {

printf(“This is the first dll src!\n”);

}

3. 2 生成动态库

gcc -O -fpic -shared -o dl.so pr1.c

4 动态库的运用 4.1 动态库的隐式调用

在编译调用库函数代码时指明动态库的方位及姓名 main.c : int main()

{

print();

return 0;

}

gcc -o tdl main.c ./dl.so

当动态库的方位或许姓名产生改动时, 程序将无法正常运转; 而动态库替代静态库的优点之一则是经过更新动态库而随时晋级库的内容.

4.2 动态库的显式调用

显式调用动态库需求四个函数的支撑,

(1)函数 dlopen 翻开动态库,

(2)函数 dlsym 获取动态库中方针基址,

(3)函数 dlerror 获取显式动态库操作中的错误信息,

(4)函数 doclose 封闭动态库. main.c :

int main()

{

void *pHandle;

void (*pFunc)(); // 指向函数的指针

int *p;

pHandle = dlopen(“./d1.so”, RTLD_NOW); // 翻开动态库

if(!pHandle) {

printf(“Can’t find d1.so \n”);

exit(1);

}

pFunc = (void (*)())dlsym(pHandle, “print”); // 获取库函数 print 的地址

if(pFunc)

{ pFunc();

} else

{ printf(“Can’t find function print\n”);

} p = (int *)dlsym(pHandle, “p”); // 获取库变量 p 的地址

if(p)

{ printf(“p = %d\n”, *p);

} else

{ printf(“Can’t find int p\n”);

} dlclose(pHandle); // 封闭动态库

return 0;

}

gcc -o tds main.c –ld1 –L. 此 时还不能当即./tds,由于在动态函数库运用时,会查找/usr/lib、/lib目录下的动态函数库,而此刻咱们生成的库不在里边。 解决办法有: (1)最直接最简略的办法便是把生成的动态链接库放到/usr/lib或/lib中去。 (2) export LDLIBRARYPATH=$(pwd) (3)别的还可以在/etc/ld.so.conf文件里参加咱们生成的库的目录,然后/sbin/ldconfig。 /etc/ld.so.conf是非常重要的一个目录,里边寄存的是链接器和加载器查找同享库时要检查的目录,默许是从/usr/lib /lib中读取的,所以想要顺畅运转,咱们也可以把咱们库的目录参加到这个文件中并履行/sbin/ldconfig 。别的还有个文件需求了解/etc/ld.so.cache,里边保存了常用的动态函数库,且会先把他们加载到内存中,由于内存的拜访速度远远大于硬盘的 拜访速度,这样可以进步软件加载动态函数库的速度了。 5 库依靠的检查 运用ldd指令来检查履行文件依靠于哪些库。 该指令用于判别某个可履行的 binary 档案含有什么动态函式库。 ldd [-vdr] [filename] 参数阐明: –version  打印ldd的版别号 -v –verbose  打印一切信息,例如包括符号的版别信息 -d –data-relocs  履行符号重布置,并陈述短少的方针方针(只对ELF格局适用) -r –function-relocs  对方针方针和函数履行从头布置,并陈述短少的方针方针和函数(只对ELF格局适用) –help 用法信息。 假如指令行中给定的库姓名包括’/’,这个程序的libc5版别将运用它作为库姓名;不然它将在规范方位查找库。运转一个当时目录下的同享库,加前缀”./”。

ps:这儿介绍的仅仅关于linux下动态库和静态库的基本知识,只需学就能会。可是个人以为库的高超之处在其接口规划的合理性和可扩展性,要契合通用的习气等等,以为 《C言语接口与完成:创立可重用软件的技能》可以带来这方面的更深的见地,当然这需求经历领会。 个人以为,好的接口除了常见的那些规范外,要害也应该让运用者以最小的价值了解并能最快的服务个人的工作中,在运用过程中逐渐的了解自己需求的部分。

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/zhishi/moni/297172.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部