青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆 - 505  文章 - 1034  trackbacks - 0
<2007年8月>
2930311234
567891011
12131415161718
19202122232425
2627282930311
2345678


子曾經曰過:編程無他,唯手熟爾!

常用鏈接

留言簿(94)

隨筆分類(649)

隨筆檔案(505)

相冊

BCB

Crytek

  • crymod
  • Crytek's Offical Modding Portal

Game Industry

OGRE

other

Programmers

Qt

WOW Stuff

搜索

  •  

積分與排名

  • 積分 - 918837
  • 排名 - 14

最新隨筆

最新評論

閱讀排行榜

評論排行榜

許多Visual C++的使用者都碰到過LNK2005:symbol already defined和LNK1169:one or more multiply defined symbols found這樣的鏈接錯誤,而且通常是在使用第三方庫時遇到的。對于這個問題,有的朋友可能不知其然,而有的朋友可能知其然卻不知其所以然,那么本文就試圖為大家徹底解開關于它的種種疑惑。

    大家都知道,從C/C++源程序到可執行文件要經歷兩個階段:(1)編譯器將源文件編譯成匯編代碼,然后由匯編器(assembler)翻譯成機器指令 (再加上其它相關信息)后輸出到一個個目標文件(object file,VC的編譯器編譯出的目標文件默認的后綴名是.obj)中;(2)鏈接器(linker)將一個個的目標文件(或許還會有若干程序庫)鏈接在一起生成一個完整的可執行文件。

    編譯器編譯源文件時會把源文件的全局符號(global symbol)分成強(strong)和弱(weak)兩類傳給匯編器,而隨后匯編器則將強弱信息編碼并保存在目標文件的符號表中。那么何謂強弱呢?編譯器認為函數與初始化了的全局變量都是強符號,而未初始化的全局變量則成了弱符號。比如有這么個源文件:

extern int errorno;
int buf[2] = {1,2};
int *p;

int main()
{
   return 0;
}

其中main、buf是強符號,p是弱符號,而errorno則非強非弱,因為它只是個外部變量的使用聲明。

    有了強弱符號的概念,我們就可以看看鏈接器是如何處理與選擇被多次定義過的全局符號:

規則1: 不允許強符號被多次定義(即不同的目標文件中不能有同名的強符號);


規則2: 如果一個符號在某個目標文件中是強符號,在其它文件中都是弱符號,那么選擇強符號;


規則3: 如果一個符號在所有目標文件中都是弱符號,那么選擇其中任意一個;

由上可知多個目標文件不能重復定義同名的函數與初始化了的全局變量,否則必然導致LNK2005和LNK1169兩種鏈接錯誤。可是,有的時候我們并沒有在自己的程序中發現這樣的重定義現象,卻也遇到了此種鏈接錯誤,這又是何解?嗯,問題稍微有點兒復雜,容我慢慢道來。


    眾所周知,ANSI C/C++ 定義了相當多的標準函數,而它們又分布在許多不同的目標文件中,如果直接以目標文件的形式提供給程序員使用的話,就需要他們確切地知道哪個函數存在于哪個目標文件中,并且在鏈接時顯式地指定目標文件名才能成功地生成可執行文件,顯然這是一個巨大的負擔。所以C語言提供了一種將多個目標文件打包成一個文件的機制,這就是靜態程序庫(static library)。開發者在鏈接時只需指定程序庫的文件名,鏈接器就會自動到程序庫中尋找那些應用程序確實用到的目標模塊,并把(且只把)它們從庫中拷貝出來參與構建可執行文件。幾乎所有的C/C++開發系統都會把標準函數打包成標準庫提供給開發者使用(有不這么做的嗎?)。

    程序庫為開發者帶來了方便,但同時也是某些混亂的根源。我們來看看鏈接器是如何解析(resolve)對程序庫的引用的。
   
在符號解析(symbol resolution)階段,鏈接器按照所有目標文件和庫文件出現在命令行中的順序從左至右依次掃描它們,在此期間它要維護若干個集合:(1)集合E是將被合并到一起組成可執行文件的所有目標文件集合;(2)集合U是未解析符號(unresolved symbols,比如已經被引用但是還未被定義的符號)的集合;(3)集合D是所有之前已被加入到E的目標文件定義的符號集合。一開始,E、U、D都是空的。

(1): 對命令行中的每一個輸入文件f,鏈接器確定它是目標文件還是庫文件,如果它是目標文件,就把f加入到E,并把f中未解析的符號和已定義的符號分別加入到U、D集合中,然后處理下一個輸入文件。

(2): 如果f是一個庫文件,鏈接器會嘗試把U中的所有未解析符號與f中各目標模塊定義的符號進行匹配。如果某個目標模塊m定義了一個U中的未解析符號,那么就把 m加入到E中,并把m中未解析的符號和已定義的符號分別加入到U、D集合中。不斷地對f中的所有目標模塊重復這個過程直至到達一個不動點(fixed point),此時U和D不再變化。而那些未加入到E中的f里的目標模塊就被簡單地丟棄,鏈接器繼續處理下一輸入文件。


(3): 如果處理過程中往D加入一個已存在的符號,或者當掃描完所有輸入文件時U非空,鏈接器報錯并停止動作。否則,它把E中的所有目標文件合并在一起生成可執行文件。

    VC帶的編譯器名字叫cl.exe,它有這么幾個與標準程序庫有關的選項: /ML、/MLd、/MT、/MTd、/MD、/MDd。這些選項告訴編譯器應用程序想使用什么版本的C標準程序庫。/ML(缺省選項)對應單線程靜態版的標準程序庫(libc.lib);/MT對應多線程靜態版標準庫(libcmt.lib),此時編譯器會自動定義_MT宏;/MD對應多線程DLL版 (導入庫msvcrt.lib,DLL是msvcrt.dll),編譯器自動定義_MT和_DLL兩個宏。后面加d的選項都會讓編譯器自動多定義一個 _DEBUG宏,表示要使用對應標準庫的調試版,因此/MLd對應調試版單線程靜態標準庫(libcd.lib),/MTd對應調試版多線程靜態標準庫 (libcmtd.lib),/MDd對應調試版多線程DLL標準庫(導入庫msvcrtd.lib,DLL是msvcrtd.dll)。雖然我們的確在編譯時明白無誤地告訴了編譯器應用程序希望使用什么版本的標準庫,可是當編譯器干完了活,輪到鏈接器開工時它又如何得知一個個目標文件到底在思念誰?為了傳遞相思,我們的編譯器就干了點秘密的勾當。在cl編譯出的目標文件中會有一個專門的區域(關心這個區域到底在文件中什么地方的朋友可以參考COFF和 PE文件格式)存放一些指導鏈接器如何工作的信息,其中有一種就叫缺省庫(default library),這些信息指定了一個或多個庫文件名,告訴鏈接器在掃描的時候也把它們加入到輸入文件列表中(當然順序位于在命令行中被指定的輸入文件之后)。說到這里,我們先來做個小實驗。寫個頂頂簡單的程序,然后保存為main.c :

/* main.c */
int main() { return 0; }


  用下面這個命令編譯main.c(什么?你從不用命令行來編譯程序?這個......) :

cl /c main.c

/c 是告訴cl只編譯源文件,不用鏈接。因為/ML是缺省選項,所以上述命令也相當于: cl /c /ML main.c 。如果沒什么問題的話(要出了問題才是活見鬼!當然除非你的環境變量沒有設置好,這時你應該去VC的bin目錄下找到vcvars32.bat文件然后運行它。),當前目錄下會出現一個main.obj文件,這就是我們可愛的目標文件。隨便用一個文本編輯器打開它(是的,文本編輯器,大膽地去做別害怕),搜索"defaultlib"字符串,通常你就會看到這樣的東西: "-defaultlib:LIBC -defaultlib:OLDNAMES"。啊哈,沒錯,這就
是保存在目標文件中的缺省庫信息。我們的目標文件顯然指定了兩個缺省庫,一個是單線程靜態版標準庫libc.lib(這與/ML選項相符),另外一個是oldnames.lib(它是為了兼容微軟以前的C/C++開發系統)。

    VC的鏈接器是link.exe,因為main.obj保存了缺省庫信息,所以可以用

link main.obj libc.lib

或者

link main.obj

來生成可執行文件main.exe,這兩個命令是等價的。但是如果你用

link main.obj libcd.lib

的話,鏈接器會給出一個警告: "warning LNK4098: defaultlib "LIBC" conflicts with use of other libs; use /NODEFAULTLIB:library",因為你顯式指定的標準庫版本與目標文件的缺省值不一致。通常來說,應該保證鏈接器合并的所有目標文件指定的缺省標準庫版本一致,否則編譯器一定會給出上面的警告,而LNK2005和LNK1169鏈接錯誤則有時會出現有時不會。那么這個有時到底是什么時候?呵呵,別著急,下面的一切正是為喜歡追根究底的你準備的。

    建一個源文件,就叫mylib.c,內容如下:

/* mylib.c */
#include

void foo()
{
   printf("%s","I am from mylib!\n");
}

cl /c /MLd mylib.c


命令編譯,注意/MLd選項是指定libcd.lib為默認標準庫。lib.exe是VC自帶的用于將目標文件打包成程序庫的命令,所以我們可以用

lib /OUT:my.lib mylib.obj

將mylib.obj打包成庫,輸出的庫文件名是my.lib。接下來把main.c改成:

/* main.c */
void foo();

int main()
{
   foo();
   return 0;
}

cl /c main.c

編譯,然后用

link main.obj my.lib

進行鏈接。這個命令能夠成功地生成main.exe而不會產生LNK2005和LNK1169鏈接錯誤,你僅僅是得到了一條警告信息:"warning LNK4098: defaultlib "LIBCD" conflicts with use of other libs; use /NODEFAULTLIB:library"。我們根據前文所述的掃描規則來分析一下鏈接器此時做了些啥。

    一開始E、U、D都是空集,鏈接器首先掃描到main.obj,把它加入E集合,同時把未解析的foo加入U,把main加入D,而且因為 main.obj的默認標準庫是libc.lib,所以它被加入到當前輸入文件列表的末尾。接著掃描my.lib,因為這是個庫,所以會拿當前U中的所有符號(當然現在就一個foo)與my.lib中的所有目標模塊(當然也只有一個mylib.obj)依次匹配,看是否有模塊定義了U中的符號。結果 mylib.obj確實定義了foo,于是它被加入到E,foo從U轉移到D,mylib.obj引用的printf加入到U,同樣地, mylib.obj指定的默認標準庫是libcd.lib,它也被加到當前輸入文件列表的末尾(在libc.lib的后面)。不斷地在my.lib庫的各模塊上進行迭代以匹配U中的符號,直到U、D都不再變化。很明顯,現在就已經到達了這么一個不動點,所以接著掃描下一個輸入文件,就是libc.lib。


鏈接器發現libc.lib里的printf.obj里定義有printf,于是printf從U移到D,而printf.obj被加入到E,它定義的所有符號加入到D,它里頭的未解析符號加入到U。鏈接器還會把每個程序都要用到的一些初始化操作所在的目標模塊(比如crt0.obj等)及它們所引用的模塊 (比如malloc.obj、free.obj等)自動加入到E中,并更新U和D以反應這個變化。事實上,標準庫各目標模塊里的未解析符號都可以在庫內其它模塊中找到定義,因此當鏈接器處理完libc.lib時,U一定是空的。最后處理libcd.lib,因為此時U已經為空,所以鏈接器會拋棄它里面的所有目標模塊從而結束掃描,然后合并E中的目標模塊并輸出可執行文件。

    上文描述了雖然各目標模塊指定了不同版本的缺省標準庫但仍然鏈接成功的例子,接下來你將目睹因為這種不嚴謹而導致的悲慘失敗。

    修改mylib.c成這個樣子:

#include

void foo()
{
   // just a test , don't care memory leak
   _malloc_dbg( 1, _NORMAL_BLOCK, __FILE__, __LINE__ );
}

其中_malloc_dbg不是ANSI C的標準庫函數,它是VC標準庫提供的malloc的調試版,與相關函數配套能幫助開發者抓各種內存錯誤。使用它一定要定義_DEBUG宏,否則預處理器會把它自動轉為malloc。繼續用

cl /c /MLd mylib.c
lib /OUT:my.lib mylib.obj

編譯打包。當再次用

link main.obj my.lib

進行鏈接時,我們看到了什么?天哪,一堆的LNK2005加上個貴為"fatal error"的LNK1169墊底,當然還少不了那個LNK4098。鏈接器是不是瘋了?不,你冤枉可憐的鏈接器了,我拍胸脯保證它可是一直在盡心盡責地照章辦事。

一開始E、U、D為空,鏈接器掃描main.obj,把它加入E,把foo加入U,把 main加入D,把libc.lib加入到當前輸入文件列表的末尾。接著掃描my.lib,foo從U轉移到D,_malloc_dbg加入到U, libcd.lib加到當前輸入文件列表的尾部。然后掃描libc.lib,這時會發現libc.lib里任何一個目標模塊都沒有定義 _malloc_dbg(它只在調試版的標準庫中存在),所以不會有任何一個模塊因為_malloc_dbg而加入E,但是每個程序都要用到的初始化模塊 (如crt0.obj等)及它們所引用的模塊(比如malloc.obj、free.obj等)還是會自動加入到E中,同時U和D被更新以反應這個變化。當鏈接器處理完libc.lib時,U只剩_malloc_dbg這一個符號。最后處理libcd.lib,發現dbgheap.obj定義了 _malloc_dbg,于是dbgheap.obj加入到E,它里頭的未解析符號加入U,它定義的所有其它符號也加入D,這時災難便來了。之前 malloc等符號已經在D中(隨著libc.lib里的malloc.obj加入E而加入的),而dbgheap.obj又定義了包括malloc在內的許多同名符號,這引發了重定義沖突,鏈接器只好中斷工作并報告錯誤。

    現在我們該知道,鏈接器完全沒有責任,責任在我們自己的身上。是我們粗心地把缺省標準庫版本不一致的目標文件(main.obj)與程序庫 (my.lib)鏈接起來,導致了大災難。解決辦法很簡單,要么用/MLd選項來重編譯main.c;要么用/ML選項重編譯mylib.c。

    在上述例子中,我們擁有庫my.lib的源代碼(mylib.c),所以可以用不同的選項重新編譯這些源代碼并再次打包。可如果使用的是第三方的庫,它并沒有提供源代碼,那么我們就只有改變自己程序的編譯選項來適應這些庫了。但是如何知道庫中目標模塊指定的默認庫呢?其實VC提供的一個小工具便可以完成任務,這就是dumpbin.exe。運行下面這個命令

dumpbin /DIRECTIVES my.lib

  然后在輸出中找那些"Linker Directives"引導的信息,你一定會發現每一處這樣的信息都會包含若干個類似"-defaultlib:XXXX"這樣的字符串,其中XXXX便代表目標模塊指定的缺省庫名。

  知道了第三方庫指定的默認標準庫,再用合適的選項編譯我們的應用程序,就可以避免LNK2005和LNK1169鏈接錯誤。喜歡IDE的朋友,你一樣可以到 "Project屬性" -> "C/C++" -> "代碼生成(code generation)" -> "運行時庫(run-time library)" 項下設置應用程序的默認標準庫版本,這與命令行選項的效果是一樣的。

posted on 2007-07-25 03:09 七星重劍 閱讀(717) 評論(0)  編輯 收藏 引用 所屬分類: PL--c/c++
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲欧美日韩国产综合在线| 一区在线电影| 久久久国产91| 亚洲欧美一区二区精品久久久| 亚洲免费av电影| 亚洲精品在线免费| 一区二区三区.www| 欧美伊人久久大香线蕉综合69| 久久国产欧美| 美女性感视频久久久| 蜜桃av一区二区三区| 欧美日韩成人一区二区三区| 国产精品乱码人人做人人爱| 久久精品色图| 欧美精品97| 国产欧美日韩精品一区| 亚洲丁香婷深爱综合| 亚洲无玛一区| 欧美一区二区三区视频在线观看| 久久精品人人做人人爽电影蜜月| 久久在线视频在线| 亚洲人成在线观看网站高清| 亚洲精品美女免费| 亚洲女同性videos| 欧美成人69av| 国产农村妇女毛片精品久久麻豆| 亚洲成人资源| 先锋影音久久久| 亚洲大胆人体视频| 亚洲欧美日韩一区在线观看| 欧美激情一区二区三区| 国产欧美日韩伦理| 欧美插天视频在线播放| 国产欧美不卡| 日韩视频一区| 老**午夜毛片一区二区三区| 亚洲天堂av图片| 麻豆精品网站| 欧美制服丝袜第一页| 久久免费国产| 日韩一二三在线视频播| 久久精品亚洲精品国产欧美kt∨| 欧美日韩和欧美的一区二区| 极品裸体白嫩激情啪啪国产精品| 亚洲人线精品午夜| 久久在线视频在线| 亚洲欧美一区二区原创| 欧美日韩一区二区三区在线视频 | 亚洲大胆美女视频| 在线播放豆国产99亚洲| 一本色道久久综合亚洲精品婷婷| 久久久欧美一区二区| 亚洲精品日韩精品| 亚洲国产精品999| 亚洲网站在线观看| 亚洲国产精品久久久久婷婷884 | 中文亚洲字幕| 欧美日韩国产经典色站一区二区三区| 亚洲第一区在线观看| 久久偷看各类wc女厕嘘嘘偷窃| 亚洲一级影院| 国产精品电影网站| 欧美午夜在线一二页| 一本一本久久| 亚洲精品影视| 国产精品wwwwww| 国产精品久久久久久久久免费樱桃 | 久久久久一区| 国产婷婷色一区二区三区| 欧美在线你懂的| 久久人人九九| 亚洲国产精品久久久久婷婷老年| 久久久精品日韩欧美| 久久精品av麻豆的观看方式| 激情婷婷久久| 美国三级日本三级久久99| 久久久精品五月天| 亚洲国产综合视频在线观看| 亚洲国产99精品国自产| 欧美精品电影| 午夜欧美理论片| 久久精品99| 国产最新精品精品你懂的| 欧美综合77777色婷婷| 男人天堂欧美日韩| 欧美jizzhd精品欧美巨大免费| 日韩视频在线观看免费| 一本在线高清不卡dvd| 亚洲午夜性刺激影院| 国产精品户外野外| 最新国产の精品合集bt伙计| 久久午夜视频| 免费观看亚洲视频大全| 在线中文字幕一区| 欧美亚洲日本网站| 亚洲国产精品t66y| 在线亚洲观看| 伊人伊人伊人久久| 99伊人成综合| 亚洲国产成人精品视频| 国产精品99久久久久久久久| 一色屋精品视频在线观看网站| 日韩亚洲一区二区| 免费在线看一区| 午夜精品婷婷| 欧美成人影音| 久久久久中文| 国产精品高潮呻吟久久| 欧美freesex8一10精品| 国产精品日韩高清| 欧美www在线| 亚洲免费av片| 久久av一区二区三区漫画| 夜夜嗨av色一区二区不卡| 久久精品国产亚洲一区二区三区| 亚洲午夜极品| 欧美韩国日本一区| 欧美激情视频网站| 久久久久一本一区二区青青蜜月| 亚洲视频精品| 欧美高清视频www夜色资源网| 久久久久久久久综合| 国产精品免费看片| 夜色激情一区二区| 9l视频自拍蝌蚪9l视频成人| 美女主播一区| 亚洲高清一区二| 欧美影院精品一区| 欧美一进一出视频| 国产精品乱码一区二区三区 | 免费成人在线视频网站| 国产精品理论片| 国产一区二区日韩精品| 久久九九有精品国产23| 国产精品日韩二区| 亚洲免费人成在线视频观看| 亚洲一级片在线看| 欧美精品一线| 亚洲乱码国产乱码精品精| 亚洲精品日产精品乱码不卡| 免费欧美在线| 亚洲高清不卡| 日韩一级成人av| 欧美黑人在线观看| 亚洲美女免费精品视频在线观看| 日韩五码在线| 欧美高清你懂得| 亚洲麻豆视频| 午夜久久久久久| 老司机成人在线视频| 亚洲系列中文字幕| 香蕉av777xxx色综合一区| 国产精品久久久久久模特| 日韩午夜在线视频| 亚洲一区二区三区国产| 在线观看视频亚洲| 国产精品视频一区二区三区| 亚洲一区在线观看免费观看电影高清| 午夜精品福利一区二区三区av| 欧美午夜视频在线观看| 亚洲一区二区在线| 久久精品国产99| 亚洲国产小视频| 欧美激情自拍| 亚洲欧美日本另类| 欧美aa国产视频| 亚洲视屏在线播放| 国产综合网站| 欧美成人日韩| 亚洲欧美日韩精品| 欧美激情一二三区| 亚洲综合不卡| 国内视频一区| 夜夜嗨一区二区| 久久久噜噜噜久久中文字免| 亚洲国产成人91精品| 欧美日韩在线视频观看| 欧美一区二区三区四区在线| 亚洲第一页在线| 亚洲尤物在线| 久久久xxx| 一区二区三区视频在线观看| 国产农村妇女毛片精品久久莱园子| 久久婷婷蜜乳一本欲蜜臀| 一区二区国产日产| 欧美成人午夜| 午夜精品一区二区三区在线视 | 国产日韩欧美高清免费| 欧美激情久久久久| 欧美专区中文字幕| 9人人澡人人爽人人精品| 欧美福利一区| 久久疯狂做爰流白浆xx| 亚洲小视频在线| 亚洲免费播放| 亚洲黄色片网站| 久久在精品线影院精品国产| 亚洲亚洲精品在线观看| 最新国产乱人伦偷精品免费网站 | 国产亚洲欧洲一区高清在线观看 |