佳木斯湛栽影视文化发展公司

主頁(yè) > 知識(shí)庫(kù) > Linux系統(tǒng)下 連接器ld鏈接順序的總結(jié)

Linux系統(tǒng)下 連接器ld鏈接順序的總結(jié)

熱門(mén)標(biāo)簽:硅谷的囚徒呼叫中心 家政服務(wù)網(wǎng)絡(luò) 電話(huà)機(jī)器人搭建 外呼系統(tǒng) 美團(tuán) 解決方案 百度競(jìng)價(jià)點(diǎn)擊價(jià)格的計(jì)算公式 服務(wù)器配置

  原來(lái)ld對(duì)于鏈接一系列的庫(kù)的順序是很敏感的,不然會(huì)報(bào)undefined referenced 的函數(shù)符號(hào)錯(cuò)誤,意思就是未找到函數(shù)定義。實(shí)際上庫(kù)是能正確打開(kāi)的。如果庫(kù)libA.a依賴(lài)于庫(kù)libB.a,那么連接器的參數(shù)應(yīng)該ln -lA -lB,   必須這樣寫(xiě)。不然就會(huì)錯(cuò)誤。

  這個(gè)看來(lái)是歷史的遺留問(wèn)題。說(shuō)白了就是,如果你確定某個(gè)庫(kù)不重要,就放到最后,也就是說(shuō),庫(kù)的加載順序是按順序進(jìn)行的,從左到右,優(yōu)先級(jí)最高,這也是遠(yuǎn)古時(shí)代機(jī)器太破的原因,內(nèi)在和硬盤(pán)包括CPU都是極其稀罕珍貴的。所以要你提前搞定哪些需要哪些不需要,一旦需要,就加載上去,不需要就不要加載上去。這樣就可以解釋上面的錯(cuò)誤了,編譯器加載到socketcomm后,發(fā)現(xiàn)還有依賴(lài)庫(kù),再往后找,木有咧,于是就錯(cuò)了。

  也就是說(shuō),被依賴(lài)的盡量放到后面,讓大家可以共用。

  這就引申同來(lái)了另外一個(gè)問(wèn)題,如果有多個(gè)庫(kù),使用了相同的函數(shù)名或者類(lèi)名,結(jié)構(gòu)體名稱(chēng)會(huì)怎么樣?

  可以使用nm和readelf、ldd等命令來(lái)查看你的庫(kù)的依賴(lài)和符號(hào)表以及導(dǎo)出的函數(shù)符號(hào)等。這樣就可以來(lái)驗(yàn)證上面的因?yàn)轫樞蛟斐傻牟煌木幾g結(jié)果。

  把測(cè)試的庫(kù)分成動(dòng)、動(dòng);靜、靜;動(dòng)靜三種情況。

  第一種情況:靜靜庫(kù)都包含有相同的名字函數(shù)


復(fù)制代碼
代碼如下:
  gcc -L./ main.c -lA -lB

  編譯報(bào)錯(cuò),在multiple definition of `XXX‘。重復(fù)定義函數(shù)。

  第二種情況:動(dòng)動(dòng)庫(kù)

  這個(gè)得看加載的順序,上面說(shuō)過(guò),從左開(kāi)始,按需加載。


復(fù)制代碼
代碼如下:
  gcc main.c ./libB.so ./libA.so


復(fù)制代碼
代碼如下:
  gcc main.c ./libA.so ./libB.so

  這樣的鏈接方式為隱式鏈接,或者說(shuō)加載時(shí)鏈接,而前面的靜態(tài)庫(kù)為編譯時(shí)連接。

  也就是說(shuō),編譯的順序確定到底調(diào)用哪個(gè)庫(kù)中的同名函數(shù)。

  第三種情況:動(dòng)動(dòng),但是是動(dòng)態(tài)加載,而不上面的靜態(tài)加載。


復(fù)制代碼
代碼如下:
  gcc main.c -ldl ./libA.so ./libB.so


復(fù)制代碼
代碼如下:
  gcc main.c -ldl ./libB.so ./libA.so

  沒(méi)有加鏈接選項(xiàng)時(shí),跟第二種情況一樣。加了L選項(xiàng)后,指定哪上加載哪個(gè)。

  libB為靜態(tài)鏈接!, libA為動(dòng)態(tài)加載.


復(fù)制代碼
代碼如下:
  gcc -L./ main.c -ldl –lB

  動(dòng)態(tài)庫(kù)的庫(kù)名顯示的加載入編譯選項(xiàng)中


復(fù)制代碼
代碼如下:
  gcc -L./ main.c -ldl -lB ./libA.so


復(fù)制代碼
代碼如下:
  gcc -L./ main.c -ldl ./libA.so -lB

  在有靜態(tài)和動(dòng)態(tài)時(shí),不把動(dòng)態(tài)庫(kù)庫(kù)名顯示加入編譯選項(xiàng),輸出是正常的,如果加進(jìn)去以靜態(tài)庫(kù)為主和link順序無(wú)關(guān)。謝謝閱讀,希望能幫到大家,請(qǐng)繼續(xù)關(guān)注腳本之家,我們會(huì)努力分享更多優(yōu)秀的文章。

標(biāo)簽:北海 南昌 韶關(guān) 防城港 臨沂 邢臺(tái) 烏蘭察布 撫州

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Linux系統(tǒng)下 連接器ld鏈接順序的總結(jié)》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話(huà)咨詢(xún)

    • 400-1100-266
    岳西县| 孟村| 彭水| 阜宁县| 乌拉特中旗| 香河县| 策勒县| 西昌市| 陈巴尔虎旗| 铅山县| 淄博市| 南投县| 沂南县| 如皋市| 安徽省| 惠东县| 济宁市| 绵竹市| 梓潼县| 中阳县| 夏河县| 香格里拉县| 乌什县| 修水县| 曲麻莱县| 汉阴县| 沁源县| 镇远县| 呈贡县| 宣武区| 泰顺县| 佛山市| 自治县| 册亨县| 香港| 新巴尔虎左旗| 延津县| 美姑县| 泸定县| 普格县| 雷州市|