新聞中心
當你在編譯代碼時,你通常要處理多個源文件。開發(fā)人員傾向于將不同的類或模塊放在不同的文件中,這樣它們可以被單獨維護,甚至可能被不同的項目使用。但當你編譯這些文件時,許多文件會被編譯成一個可執(zhí)行文件。

為樺南等地區(qū)用戶提供了全套網(wǎng)頁設計制作服務,及樺南網(wǎng)站建設行業(yè)解決方案。主營業(yè)務為成都做網(wǎng)站、成都網(wǎng)站設計、成都外貿(mào)網(wǎng)站建設、樺南網(wǎng)站設計,以傳統(tǒng)方式定制建設網(wǎng)站,并提供域名空間備案等一條龍服務,秉承以專業(yè)、用心的態(tài)度為用戶提供真誠的服務。我們深信只要達到每一位用戶的要求,就會得到認可,從而選擇與我們長期合作。這樣,我們也可以走得更遠!
這通常是通過創(chuàng)建共享庫來完成的,然后從可執(zhí)行文件中動態(tài)鏈接回它們。這樣可以通過保持模塊化功能的外部性來保持可執(zhí)行文件的小型化,并確保庫可以獨立于使用它們的應用而被更新。
在編譯過程中定位一個共享對象
當你 用 GCC 編譯? 時,你通常需要在你的工作站上安裝一個庫,以便 GCC 能夠定位到它。默認情況下,GCC 假定庫在系統(tǒng)庫路徑中,例如 /lib64? 和 /usr/lib64。然而,如果你要鏈接到一個你自己的尚未安裝的庫,或者你需要鏈接到一個沒有安裝在標準位置的庫,那么你必須幫助 GCC 找到這些文件。
有兩個選項對于在 GCC 中尋找?guī)旌苤匾?/p>
- -L(大寫字母 L)在 GCC 的搜索位置上增加一個額外的庫路徑。
- -l(小寫字母 L)設置你要鏈接的庫的名字。
例如,假設你寫了一個叫做 libexample.so? 的庫,并且你想在編譯你的應用 demo.c? 時使用它。首先,從 demo.c 創(chuàng)建一個對象文件:
$ gcc -I ./include -c src/demo.c
-I? 選項在 GCC 搜索頭文件的路徑中增加了一個目錄。在這個例子中,我假設自定義頭文件在一個名為 include? 的本地目錄中。-c 選項防止 GCC 運行鏈接器,因為這個任務只是為了創(chuàng)建一個對象文件。結(jié)果如下:
$ ls
demo.o include/ lib/ src/
現(xiàn)在你可以使用 -L 選項為你的庫設置一個路徑,然后進行編譯:
$ gcc -L`pwd`/lib -o myDemo demo.o -lexample
注意,-L? 選項在 -l? 選項之前。這很重要,因為如果在你告訴 GCC 查找非默認庫之前沒有將 -L 添加到 GCC 的搜索路徑中,GCC 就不知道要在你的自定義位置上搜索。編譯成功了,但當你試圖運行它時,卻出現(xiàn)了問題:
$ ./myDemo
./myDemo: error while loading shared libraries:
libexample.so: cannot open shared object file:
No such file or directory
用 ldd 排除故障
ldd 工具可以打印出共享對象的依賴關系,它在排除類似問題時很有用:
$ ldd ./myDemo
linux-vdso.so.1 (0x00007ffe151df000)
libexample.so => not found
libc.so.6 => /lib64/libc.so.6 (0x00007f514b60a000)
/lib64/ld-linux-x86-64.so.2 (0x00007f514b839000)
你已經(jīng)知道定位不到 libexample?,但 ldd? 輸出至少確認了它對工作庫的期望位置。例如,libc.so.6?已經(jīng)被定位,ldd 顯示其完整路徑。
LD_LIBRARY_PATH
LD_LIBRARY_PATH 環(huán)境變量? 定義了庫的路徑。如果你正在運行一個依賴于沒有安裝到標準目錄的庫的應用程,你可以使用 LD_LIBRARY_PATH 添加到系統(tǒng)的庫搜索路徑。
有幾種設置環(huán)境變量的方法,但最靈活的是在運行命令前放置環(huán)境變量??纯丛O置 LD_LIBRARY_PATH? 對 ldd 命令在分析一個“損壞”的可執(zhí)行文件時的作用:
$ LD_LIBRARY_PATH=`pwd`/lib ldd ./
linux-vdso.so.1 (0x00007ffe515bb000)
libexample.so => /tmp/Demo/lib/libexample.so (0x0000...
libc.so.6 => /lib64/libc.so.6 (0x00007eff037ee000)
/lib64/ld-linux-x86-64.so.2 (0x00007eff03a22000)
這也同樣適用于你的自定義命令:
$ LD_LIBRARY_PATH=`pwd`/lib myDemo
hello world!
然而,如果你移動庫文件或可執(zhí)行文件,它又會失效:
$ mv lib/libexample.so ~/.local/lib64
$ LD_LIBRARY_PATH=`pwd`/lib myDemo
./myDemo: error while loading shared libraries...
要修復它,你必須調(diào)整 LD_LIBRARY_PATH 以匹配庫的新位置:
$ LD_LIBRARY_PATH=~/.local/lib64 myDemo
hello world!
何時使用 LD_LIBRARY_PATH
在大多數(shù)情況下,LD_LIBRARY_PATH? 不是你需要設置的變量。按照設計,庫安裝到 /usr/lib64? 中,因此應用自然會在其中搜索所需的庫。在兩種情況下,你可能需要使用 LD_LIBRARY_PATH:
- 你正在編譯的軟件需要鏈接到本身剛剛編譯但尚未安裝的庫。良好設計的構(gòu)建系統(tǒng),例如Autotools? 和CMake,可以幫助處理這個問題。
- 你正在使用設計為在單個目錄之外運行的軟件,它沒有安裝腳本,或安裝腳本將庫放置在非標準目錄中。一些應用具有 Linux 用戶可以下載、復制到/opt? 并在“不安裝”的情況下運行的版本。LD_PATH_LIBRARY 變量是通過封裝腳本設置的,因此用戶通常甚至不知道它已被設置。
編譯軟件為你在運行系統(tǒng)方面提供了很大的靈活性。LD_LIBRARY_PATH? 變量以及 -L? 和 -l GCC 選項是這種靈活性的組成部分。
名稱欄目:編譯代碼時動態(tài)地鏈接庫
URL網(wǎng)址:http://www.5511xx.com/article/cdpsjjh.html


咨詢
建站咨詢
