制作和使用Linux系统下的静态链接库文件(.a)和动态链接库文件(.so)

一:静态链接库

(1):制作静态链接库文件(.a)

    gcc    -c         ysq.c      -o     ysq.o 
    ar       -rc        libysq.a    ysq.o

1>gcc 在编译时加上-c 选项,只编译不链接,生成.o的目标文件;

2>然后使用ar工具进行打包成.a归档文件
 3>库名不能随便乱起,一般是lib+库名称,后缀名是.a表示是一个归档文件
注意:制作出来了静态库之后,发布时需要发布.a文件和.h文件。

(2):使用静态链接库文件

编译链接程序时如果未详细告知链接器库文件的相关信息,可能链接不成功,链接器找不到库,链接器不错。

把.a和.h都放在我引用的文件夹下(我一般情况下放在自己项目中使用库的源文件在同一个目录下),然后在.c文件(使用库的文件)中包含库的.h,然后直接使用库函数。以下面的描述为例,其中func1,fun2是.a 静态链接库中定义实现的函数,test.c 是使用库的源文件
第一次,编译方法: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 -lysq
报错信息:/usr/bin/ld: cannot find -lysq
collect2: error: ld returned 1 exit status

尽管我们通过-lysq 的方式已经告诉了链接器要链接的库名,链接器还是找不到以ysq为库名的库,原因是 libysq.a 是一个外来库,不是编译系统里自带的库。


第三次,编译方法:gcc test.c -o test -lysq -L.
无报错,生成test,链接成功。

其中的-lysq, 是告诉链接器要链接libysq.a或libysq.so,    -L. 是告诉链接器所要链接的库的路径就在当前目录下。当链接器找不到库时,我们就要显示的告知链接器所要链接的库的相关信息,让链接器能够找到这个外来库。

-lXXX,其中XXX是所要链接的库名,-LXXX,其中XXX是所要链接的库文件在当前Linux系统下的路径。

补充一下:我们在拿到一个.a 文件后,可以使用nm 命令获取.a 文件中的一些信息,比如当前库中有哪些.o文件和各个目标文件(.o)中都有哪些函数。直接使用在Linux 命令行下键入 nm XXX.a 即可

二、动态链接库

(1):制作动态链接库文件(.so)。
    gcc   ysq.c     -o   ysq.o   -c   -fPIC
    gcc   -o   libysq.so ysq.o  -shared 
        -fPIC是位置无关码,-shared是按照共享库的方式来链接。
注意:做库的人给用库的人发布库时,发布libxxx.so和xxx.h即可。

(2):使用动态链接库

test.c 是使用库的文件或测试库的文件

第一次,编译方法: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 -lysq
报错信息:/usr/bin/ld: cannot find -lysq
collect2: error: ld returned 1 exit status

第三次,编译方法:gcc test.c -o test -lysq -L.
编译链接成功

但是运行test出错,报错信息:
error while loading shared libraries: libysq.so: cannot open shared object file: No such file or directory

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

解决方法一:
将libysq.so放到固定目录下就可以了,这个固定目录一般是/usr/lib目录,这个目录是编译系统所维护的库目录。
cp     libysq.so   /usr/lib    即可

解决方法二:使用环境变量LD_LIBRARY_PATH。操作系统在加载固定目录/usr/lib里的库之前,会先去LD_LIBRARY_PATH这个环境变量所指定的目录下去寻找,如果找到了,就不用去/usr/lib下面找了,如果没找到再去/usr/lib下面找。所以解决方案就是将libysq.so所在的目录导出到环境变量LD_LIBRARY_PATH中即可。
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/mnt/hgfs/Winshare/s5pv210/AdvancedC/4.6.PreprocessFunction/4.6.12.sharedobject.c/sotest
 



 

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值