C语言高级专题(8)------- 静态库和动态库的制作与使用

前言:


函数库的提供形式:动态链接库与静态链接库


(1)早期的函数共享都是以源代码的形式进行的。这种方式共享是最彻底的(后来这种源码共享的方向就形成了我们现在的开源社区)。但是这种方式有它的缺点,缺点就是无法以商业化形式来发布函数库。
(2)商业公司需要将自己的有用的函数库共享给被人(当然是付费的),但是又不能给客户源代码。这时候的解决方案就是以库(主要有2种:静态库和动态库)的形式来提供。

(3)比较早出现的是静态链接库。静态库其实就是商业公司将自己的函数库源代码经过只编译不连接形成.o的目标文件,然后用ar工具将.o文件归档成.a的归档文件(.a的归档文件又叫静态链接库文件)。商业公司通过发布.a库文件和.h头文件来提供静态库给客户使用;客户拿到.a和.h文件后,通过.h头文件得知库中的库函数的原型,然后在自己的.c文件中直接调用这些库文件,在连接的时候链接器会去.a文件中拿出被调用的那个函数的编译后的.o二进制代码段链接进去形成最终的可执行程序。

(4)动态链接库比静态链接库出现的晚一些,效率更高一些,是改进型的。现在我们一般都是使用动态库。静态库在用户链接自己的可执行程序时就已经把调用的库中的函数的代码段链接进最终可执行程序中了,这样好处是可以执行,坏处是太占地方了。尤其是有多个应用程序都使用了这个库函数时,实际上在多个应用程序最后生成的可执行程序中都各自有一份这个库函数的代码段。当这些应用程序同时在内存中运行时,实际上在内存中有多个这个库函数的代码段,这完全重复了。而动态链接库本身不将库函数的代码段链接入可执行程序,只是做个标记。然后当应用程序在内存中执行时,运行时环境发现它调用了一个动态库中的库函数时,会去加载这个动态库到内存中,然后以后不管有多少个应用程序去调用这个库中的函数都会跳转到第一次加载的地方去执行(不会重复加载)。
在这里插入图片描述


一,制作静态链接库并使用



1.制作静态链接库


(1)写一个aston.c文件:简单的实现两个函数

#include <stdio.h>

void func1(void)
{
	printf("func1 in aston.c.\n");
}


int func2(int a, int b)
{
	printf("func2 in aston.c.\n");
	return a + b;
}

(2)再写一个aston.h文件:声明两个函数

void func1(void);
int func2(int a, int b);

(3)然后写一个makefile来进行编译生成静态库


all:
	gcc aston.c -o aston.o -c
	ar -rc libaston.a aston.o

makefile分析:
在这里插入图片描述
(4)编译过程:
在这里插入图片描述
注意:使用gcc -c只编译不连接,生成.o文件;然后使用ar工具进行打包成.a归档文件。库名不能随便乱起,一般是lib+库名称,后缀名是.a表示是一个归档文件制作出来了静态库之后,发布时需要发布.a文件和.h文件


2.使用静态链接库


(1)新建test文件夹下写一个test.c,测试使用静态库

#include "aston.h"
#include <stdio.h>


int main(void)
{
	func1();
	
	int a = func2(4, 5);
	printf("a = %d.\n", a);
	
	return 0;
}

(2)把.a和.h都放在我引用的文件夹test下,然后在.c文件中包含库的.h,然后直接使用库函数。
在这里插入图片描述
(3)编译方法:gcc test.c -o test -laston -L.,生成test,执行正确。
在这里插入图片描述

(3)除了ar名另外,还有个nm命令也很有用,它可以用来查看一个.a文件中都有哪些符号
在这里插入图片描述


二,制作动态链接库并使用


动态链接库的后缀名是.so(对应windows系统中的dll),静态库的扩展名是.a


1.创建一个动态链接库


(1)同样的先写一个aston.c文件:简单的实现两个函数

#include <stdio.h>

void func1(void)
{
	printf("func1 in aston.c.\n");
}


int func2(int a, int b)
{
	printf("func2 in aston.c.\n");
	return a + b;
}

(2)再写一个aston.h文件:声明两个函数

void func1(void);
int func2(int a, int b);

(3)然后写一个makefile来进行编译生成静态库

	gcc aston.c -o aston.o -c -fPIC
	gcc -o libaston.so aston.o -shared 
	-fPIC是位置无关码,-shared是按照共享库的方式来链接。

注意:做库的人给用库的人发布库时,发布libxxx.so和xxx.h即可。


2.使用自己创建的共享库


(1)新建test文件夹下写一个test.c,测试使用静态库

#include "aston.h"
#include <stdio.h>


int main(void)
{
	func1();
	
	int a = func2(4, 5);
	printf("a = %d.\n", a);
	
	return 0;
}

(2)把.a和.h都放在我引用的文件夹test下,然后在.c文件中包含库的.h,然后直接使用库函数。
在这里插入图片描述

(3)编译方法:gcc test.c -o test -laston -L.编译成功

在这里插入图片描述
但是运行出错,报错信息:

error while loading shared libraries: libaston.so: cannot open shared object file: No such file or directory

错误原因动态链接库运行时需要被加载(运行时环境在执行test程序的时候发现他动态链接了libaston.so,于是乎会去固定目录尝试加载libaston.so,如果加载失败则会打印以上错误信息。)

解决方法一
将libaston.so放到固定目录下就可以了,这个固定目录一般是/usr/lib目录。
cp libaston.so /usr/lib即可
在这里插入图片描述

解决方法二:使用环境变量LD_LIBRARY_PATH。操作系统在加载固定目录/usr/lib之前,会先去LD_LIBRARY_PATH这个环境变量所指定的目录下去寻找,如果找到就不用去/usr/lib下面找了,如果没找到再去/usr/lib下面找。所以解决方案就是将libaston.so所在的目录导出到环境变量LD_LIBRARY_PATH中即可。

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/mnt/hgfs/winshare/code/archiveuse/test

执行结果:
在这里插入图片描述

ldd命令:作用是可以在一个使用了共享库的程序执行之前解析出这个程序使用了哪些共享库,并且查看这些共享库是否能被找到,能被解析(决定这个程序是否能正确执行)。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值