制作动态链接库并使用
函数库是通过头文件声明的函数原型,调用生成的链接库文件(.a.so.dll)。
无需提供源码。
动态链接库与静态链接库
动态链接库程序小,每次需要调用时访问库文件。依赖库运行。
静态链接库程序大,是将相关库文件函数直接嵌入代码中。编译完成后不依赖库。
编译调用库函数的C代码时,需要链接库
比如使用math.h中函数的代码编译时
gcc abc.c -o app -lm//链接时可以用-lxxx来指示链接器去到libxxx.so中去查找这个函数。
1、自己制作静态链接库并使用
(1)第一步:自己制作静态链接库
首先使用gcc -c只编译不连接,生成.o文件;然后使用ar工具进行打包成.a归档文件
库名不能随便乱起,一般是lib+库名称,后缀名是.a表示是一个归档文件
注意:制作出来了静态库之后,发布时需要发布.a文件和.h文件。
(2)第二步:使用静态链接库
把.a和.h都放在我引用的文件夹下,然后在.c文件中包含库的.h,然后直接使用库函数。
第一次,编译方法:gcc Song.c -o app
报错信息:test.c:(.text+0xa): undefined reference to `Song'
第二次,编译方法:gcc test.c -o test -laston
报错信息:/usr/bin/ld: cannot find -laston
collect2: error: ld returned 1 exit status
第三次,编译方法:gcc Song.c -o app -laston -L.
无报错,生成test,执行正确。
(3)除了ar名另外,还有个nm命令也很有用,它可以用来查看一个.a文件中都有哪些符号
Makefile
all: gcc Song.c -o Song.o -c ar -rc libSong.a Song.o
Song.c
#include<stdio.h> #include"Song.h" void Song() { printf("宋鹏飞\n"); } int Songpengfei(int a) { int i=0; for(i=0;i<a;i++) { printf("*"); } printf("\n"); }
Song.h
void Song(); int Songpengfei(int a);
2、自己制作动态链接库并使用
代码同上
Makefile
all: gcc Song.c -o Song.o -c -fPIC gcc -o libSong.so Song.o -shared
(1)动态链接库的后缀名是.so(对应windows系统中的dll),静态库的扩展名是.a
(2)第一步:创建一个动态链接库。
gcc Song.c -o Song.o -c -fPIC
gcc -o libSong.so Song.o -shared
-fPIC是位置无关码,-shared是按照共享库的方式来链接。
注意:做库的人给用库的人发布库时,发布libxxx.so和xxx.h即可。
(3)第二步:使用自己创建的共享库。
第一步,编译方法:gcc Song.c -o app
报错信息:test.c:(.text+0xa): undefined reference to `Song'
collect2: error: ld returned 1 exit status
第二步,编译方法:gcc Song.c -o app -lSong
报错信息:/usr/bin/ld: cannot find -lSong
collect2: error: ld returned 1 exit status
第三步,编译方法:gcc Song.c -o app -lSong -L.
编译成功
但是运行出错,报错信息:
error while loading shared libraries: libSong.so: cannot open shared object file: No such file or directory
错误原因:动态链接库运行时需要被加载(运行时环境在执行test程序的时候发现他动态链接了libSong.so,于是乎会去固定目录尝试加载libSong.so,如果加载失败则会打印以上错误信息。)
解决方法一:
将libSong.so放到固定目录下就可以了,这个固定目录一般是/usr/lib目录。
cp libSong.so /usr/lib即可
解决方法二:使用环境变量LD_LIBRARY_PATH。操作系统在加载固定目录/usr/lib之前,会先去LD_LIBRARY_PATH这个环境变量所指定的目录下去寻找,如果找到就不用去/usr/lib下面找了,如果没找到再去/usr/lib下面找。所以解决方案就是将libSong.so所在的目录导出到环境变量LD_LIBRARY_PATH中即可。
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/mnt/hgfs/linuxGX/c/D动态链接库/cs
在ubuntu中还有个解决方案三,用ldconfig
(4)ldd命令:作用是可以在一个使用了共享库的程序执行之前解析出这个程序使用了哪些共享库,并且查看这些共享库是否能被找到,能被解析(决定这个程序是否能正确执行)。