解決undefined reference 問題的方法

       最近把開發板上的程序移植到目標板上,編譯通過之後。鏈接時,一堆的... undefined reference ...       經過大半天的分析,終於都搞定了。在這裏做下總結,有不對的地方還請多多指教

 

-------------------------------------------------

引起此類問題的原因主要有以下幾類:

 

1.      缺少某個源文件,或源文件中缺少某個函數

2.      源文件中的宏,未正確開啓

3.      鏈接時,多個庫的順序未根據依賴關係安排

4.      編譯時的先後順序未按依賴關係安排

5.      c++引用c中定義的函數,沒有在c函數聲明時添加 extern "C"。導致c++編譯器不能對其進行正確的解析

 

     

前兩類問題比較好解決,一般把未添加的函數,源文件或者庫添加到編譯環境中。可以通過兩種方式來實現修正。

1. 使用圖形化的IDE(SourceInsight, Eclipse等),找到未定義變量或函數所需的文件和庫。

2. Linux下使用grep命令也可以快速的找到,如可以在項目(庫)目錄下,使用 grep -n 'something' *

  

第三類和第四類問題,主要是解決鏈接順序。 如果liba依賴於libb,在Makefile的LD選項中,libb應出現在liba的左側。

因爲鏈接是從左向右進行。obj文件中的UND類型的符號會在和庫進行鏈接時進行匹配,如果匹配成功鏈接器會將這個符號

放到可執行程序的符號表中。如果所有需鏈接的庫都鏈接完成後,還有UND類型的符號表,鏈接器報undefined reference的錯誤。    

具體可以參考《深入理解計算機系統》中的鏈接一章。

 

在查找鏈接錯誤時,readelf , objdump等工具可以提供一些幫助。可以通過它們發現究竟是鏈接順序的問題還是在庫中真的不存在

某個指定的符號。如查看liba.a中的符號可以通過以下命令查看(可執行文件和動態庫亦可)

      readelf -s     liba.a

      objdump -t   liba.a

 

對第五類的問題可以直接將c中extern的函數包含在以下宏之間

#ifdef __cplusplus

extern "C" {

#endif

 

 

#ifdef __cplusplus

}

#endif

-------------------End-----------------------

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章