g++編譯后運行時無法鏈接動態庫的解決方法:
問題發現:
$ g++ -Wall -o hellobrowser.exec hellobrowser.c -I/usr/local/include -L/usr/local/lib -lmicrohttpd -ldl
$ ./hellobrowser.exec
error while loading shared libraries: libmicrohttpd.so.10: cannot open shared object file: No such file or directory
$ whereis libmicrohttpd
發現/usr/local/lib下有這個庫
$ cd /usr/local/lib
$ find libmicrohttpd.so.10
的確有這個文件。
這是因為動態庫默認只會在/usr/lib下找。不會去/usr/local/lib找。
解決方法:
法1:sudo vi /etc/ld.so.conf
添加一行:
/usr/local/lib
然后sudo ldconfig使得生效。這樣動態庫就會去/usr/local/lib找了。
法2:
g++ -Wall -o hellobrowser.exec hellobrowser.c -I/usr/local/include -L/usr/local/lib -lmicrohttpd -ldl -Wl,-R /usr/local/lib
法3:
$ ln -s /usr/local/lib/libmicrohttpd.so.10 /usr/lib/libmicrohttpd.so.10
# ln -s /usr/local/lib/libmicrohttpd.so /usr/lib/libmicrohttpd.so
推薦法1和法2.
分析解釋:
默認情況下,編譯器只會使用/lib和/usr/lib這兩個目錄下的庫文件。
雖然g++ -L/usr/local/lib雖然能讓鏈接器找到庫進行鏈接,但是運行時鏈接器卻找不到這個庫,如果要讓軟件運行時庫文件的路徑也得到擴展,那么我們需要增加這個庫路徑加上-Wl,-R,即添加g++參數“-Wl,-R 動態庫庫路徑”。
或者修改系統文件/etc/ld.so.conf,這個文件記錄了編譯時使用的動態鏈接庫的路徑。
補充@20170417:
如果我要指定鏈接一個具體路徑下的庫,可以這樣(就不要-l了):
g++ <其它的編譯參數,源文件,略> libx.so
