之前一段时间在学习《C++网络编程》(卷一),将书中的代码敲出来进行测试,但是却出现了使用共享库(.so)的错误。
一。错误描述
编译ACE模块后,生成的libACE.so (libACE.so -> libACE.so.6.1.6)被放置在了/usr/local/lib/目录下,相关头文件放置在/usr/local/include。我的测试文件名为:test_message_block.c,放置在/root/ace_workspace/目录下。
编译test_message_block.c:
# gcc -c test_message_block.c -o test_message_block -I /usr/local/include -L /usr/local/lib/ -lACE
上面编译成功,生成可执行程序 test_message_block。接着运行它,却出现错误。
“./test_message_block: error while loading shared libraries: libACE.so.6.1.6: cannot open shared object file: No such file or directory”
二。错误分析
看到上面的错误,当时就觉得都已经生成可执行程序,怎么会出现找不到共享库呢?而且,在编译的时候已经提供了路径和文件了?
对着这个问题一时搞不清楚,就拷贝了错误信息在网上找了一下,同时看了一下《GNU_linux编程指南(第二版)》的第十章,学习了一下Linux中库的使用。
通过学习,了解到除了在链接时需要使用共享库外,在运行时也是需要共享库的。在链接的时候,连接器(linker)只是将共享库的引用(符号)放到了可执行程序中。所以在运行时,有一个加载器 ld.so 再将可执行程序中引用(符号)链接到适当的共享库中。出现上面错误的原因就是因为ld.so找不到libACE.so.6.1.6。而在linker链接的时候,我们提供的库路径对于ld.so是未知。ld.so是按照自己定义的路径来查找库文件的,当然我们可以修改ld.so查找的路径。
三。错误修改
现在知道了出现上面错误的原因是因为ld.so找不到库文件,那为了解决这个问题,就需要告诉ld.so去哪里能找到我们的库文件libACE.so.6.1.6。
首先,我们需要知道ld.so一般的搜索路径是$LDLIBRARY_PATH, /etc/ld.so.cache, /usr/lib, /lib。 libACE.so.6.1.6并没有放在/usr/lib和/lib下,所以就属于一个非标准位置。而运行时使用非标准位置的库的方法有三种:
(1)设置$LD_LIBRARY_PATH=库所在的目录(多个目录以:分隔)。
(2)将库路径加入/etc/ld.so.conf或在/etc/ld.so.conf.d中创建特定的.conf文件,然后运行ldconfig更行/etc/ld.so.cache。
(3)把需要的库copy到/usr/lib或/lib。(不推荐)
这里采用方法(1),
#$LD_LIBRARY_PATH=/usr/local/lib ./test_message_block
(完)