C基础之自制静态库,动态库
前言 C基础文章连载
1.自己制作静态链接库并使用
发布版本包含.a文件和.h头文件。.c文件不会给别人
我们选择写一个makefile,这样可以将编译的过程记录下来。
all:
gcc richard.c -o richard.o -c
ar -rc librichard.a richard.o
(1)第一步:自己制作静态链接库
首先使用gcc -c只编译不连接,生成.o文件;
然后使用ar工具进行打包成.a归档文件
库名不能随便乱起,一般是lib+库名称,后缀名是.a表示是一个归档文件
注意:制作出来了静态库之后,发布时需要发布.a文件和.h文件。
将要发布的文件打包到testlib文件夹中,此文件夹是用户方自己写的文件和提供方发布的文件,类似于购买软件,自己写个.c文件,然后在商家手中买来.h和.a文件,可以在.c文件中使用购买的文件。
(2)第二步:使用静态链接库
把.a和.h都放在我引用的文件夹下,然后在.c文件中包含库的.h,然后直接使用库函数。
第一次,编译方法:gcc test.c -o test
报错信息:test.c:(.text+0xa): undefined reference to func1' test.c:(.text+0x1e): undefined reference to
func2’
第二次,编译方法:gcc test.c -o test -lrichard
报错信息:/usr/bin/ld: cannot find -lrichard
collect2: error: ld returned 1 exit status
第三次,编译方法:gcc test.c -o test -lrichard -L. //-l告诉连接器在哪个库中找,后面跟库的名字。-L就是告诉连接器到哪个目录去找,.表示在当前目录,所以-L.表示在当前目录找。
无报错,生成test,执行正确。
(3)除了ar,命令外,还有个nm命令也很有用,它可以用来查看一个.a文件中都有哪些符号
动/静态链接库制作源文件:https://download.csdn.net/download/weixin_44705391/15491242
2.自己制作动态链接库并使用
(1)动态链接库的后缀名是.so(对应windows系统中的dll),静态库的扩展名是.a
(2)第一步:创建一个动态链接库。
gcc richard.c -o richard.o -c -fPIC
gcc -o librichard.so richard.o -shared
(position independent code位置无关码,因为动态链接库都是位置无关码)
-fPIC是位置无关码,-shared是按照共享库(动态链接库)的方式来链接。
注意:做库的人给用库的人发布库时,发布libxxx.so和xxx.h即可。
(3)第二步:使用自己创建的共享库。
第一步,编译方法:gcc test.c -o test
报错信息:test.c:(.text+0xa): undefined reference to func1' test.c:(.text+0x1e): undefined reference to
func2’
collect2: error: ld returned 1 exit status
第二步,编译方法:gcc test.c -o test -lrichard
报错信息:/usr/bin/ld: cannot find -lrichard
collect2: error: ld returned 1 exit status
第三步,编译方法:gcc test.c -o test -lrichard -L.
编译成功
但是运行出错,报错信息:
error while loading shared libraries: librichard.so: cannot open shared object file: No such file or directory
错误原因:动态链接库运行时需要被加载(运行时环境在执行test程序的时候发现他动态链接了librichard.so,于是乎会去固定目录尝试加载librichard.so,如果加载失败则会打印以上错误信息。)
解决方法一:
将librichard.so放到固定目录下就可以了,这个固定目录一般是/usr/lib目录
查看/etc目录中关于so的文件:ls *so*,可看到ld.so.conf,内容是:
include /etc/ld.so.conf.d/*.conf
那么关于so的配置文件一定在ld.so.conf.d目录中,进入目录,查看libc.conf 内容如下:
/usr/local/lib
可得我们的lib文件可以放在这里。
解决方法二:
使用环境变量LD_LIBRARY_PATH。操作系统在加载固定目录/usr/lib之前,会先去LD_LIBRARY_PATH这个环境变量所指定的目录下去寻找,如果找到就不用去/usr/lib下面找了,如果没找到再去/usr/lib下面找。所以解决方案就是将librichard.so所在的目录导出到环境变量LD_LIBRARY_PATH中即可。
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/mnt/hgfs/Winshare/s5pv210/AdvancedC/PreprocessFunction/12.sharedobject.c/sotest
可以用echo $LD_LIBRARY_PATH来查看导出的目录
在ubuntu中还有个解决方案三,用ldconfig
(4)ldd命令:作用是可以在一个使用了共享库的程序执行之前解析出这个程序使用了哪些共享库,并且查看这些共享库是否能被找到,能被解析(决定这个程序是否能正确执行)。
ldd后得到类似如下结果:
linux-gate.so.1 => (0xb770f000)
librichard.so (0xb770b000)
libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb7546000)
/lib/ld-linux.so.2 (0xb7710000)
可以看到librichard.so是可以连接到的
如果将librichard.so所在的目录导出到环境变量LD_LIBRARY_PATH中,命令如下:export LD_LIBRARY_PATH= 则ldd得到结果:
linux-gate.so.1 => (0xb7710000)
librichard.so => not found
libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb754a000)
/lib/ld-linux.so.2 (0xb7711000)
librichard.so连接未找到文件。