Linux下連接器ld鏈接如何排序
Linux下連接器ld鏈接如何排序
Linux系統(tǒng)中連接器ld鏈接的順序是很重要的,如果順序不當(dāng)就會(huì)導(dǎo)致undefined referenced 函數(shù)報(bào)錯(cuò),下面學(xué)習(xí)啦小編就給大家介紹下Linux下如何排列連接器ld的鏈接順序。
原來ld對于鏈接一系列的庫的順序是很敏感的,不然會(huì)報(bào)undefined referenced 的函數(shù)符號(hào)錯(cuò)誤,意思就是未找到函數(shù)定義。實(shí)際上庫是能正確打開的。如果庫libA.a依賴于庫libB.a,那么連接器的參數(shù)應(yīng)該ln -lA -lB, 必須這樣寫。不然就會(huì)錯(cuò)誤。
這個(gè)看來是歷史的遺留問題。說白了就是,如果你確定某個(gè)庫不重要,就放到最后,也就是說,庫的加載順序是按順序進(jìn)行的,從左到右,優(yōu)先級(jí)最高,這也是遠(yuǎn)古時(shí)代機(jī)器太破的原因,內(nèi)在和硬盤包括CPU都是極其稀罕珍貴的。所以要你提前搞定哪些需要哪些不需要,一旦需要,就加載上去,不需要就不要加載上去。這樣就可以解釋上面的錯(cuò)誤了,編譯器加載到socketcomm后,發(fā)現(xiàn)還有依賴庫,再往后找,木有咧,于是就錯(cuò)了。
也就是說,被依賴的盡量放到后面,讓大家可以共用。
這就引申同來了另外一個(gè)問題,如果有多個(gè)庫,使用了相同的函數(shù)名或者類名,結(jié)構(gòu)體名稱會(huì)怎么樣?
可以使用nm和readelf、ldd等命令來查看你的庫的依賴和符號(hào)表以及導(dǎo)出的函數(shù)符號(hào)等。這樣就可以來驗(yàn)證上面的因?yàn)轫樞蛟斐傻牟煌木幾g結(jié)果。
把測試的庫分成動(dòng)、動(dòng);靜、靜;動(dòng)靜三種情況。
第一種情況:動(dòng)動(dòng)庫
這個(gè)得看加載的順序,上面說過,從左開始,按需加載。
復(fù)制代碼代碼如下:
gcc main.c 。/libB.so 。/libA.so
復(fù)制代碼代碼如下:
gcc main.c 。/libA.so 。/libB.so
這樣的鏈接方式為隱式鏈接,或者說加載時(shí)鏈接,而前面的靜態(tài)庫為編譯時(shí)連接。
也就是說,編譯的順序確定到底調(diào)用哪個(gè)庫中的同名函數(shù)。
第二種情況:靜靜庫都包含有相同的名字函數(shù)
復(fù)制代碼代碼如下:
gcc -L./ main.c -lA -lB
編譯報(bào)錯(cuò),在multiple definition of `XXX‘。重復(fù)定義函數(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
沒有加鏈接選項(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)庫的庫名顯示的加載入編譯選項(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)庫庫名顯示加入編譯選項(xiàng),輸出是正常的,如果加進(jìn)去以靜態(tài)庫為主和link順序無關(guān)。
上面就是Linux系統(tǒng)下連接器ld鏈接順序的介紹了,不同的順序就會(huì)造成不同的編譯結(jié)果,因此被依賴的盡量放到后面,你記住了嗎?