C++中的“error:LNK2005 已經在*.obj中定義”異常問題
異常現象如下:

“error LNK2005: 已經在*.obj中定義”
編程中經常能遇到LNK2005錯誤——重復定義錯誤,其實LNK2005錯誤並不是一個很難解決的錯誤,弄清楚它形成的原因,就可以輕松解決它了。
1. 造成LNK2005錯誤主要有以下幾種情況:
(1)重復定義全局變量。可能存在兩種情況:
【1】對於一些初學編程的程序員,有時候會以為需要使用全局變量的地方就可以使用定義申明一下。其實這是錯誤的,全局變量是針對整個工程的。
正確的應該是在一個CPP文件中定義如下:
int g_Test;
那么在使用的CPP文件中就應該使用:
extern int g_Test
即可,如果還是使用int g_Test,那么就會產生LNK2005錯誤,一般錯誤錯誤信息類似:
“*.obj error LNK2005 int book c? already defined in *.obj”
切記的就是不能給變量賦值否則還是會有LNK2005錯誤。
這里需要的是“聲明”,不是“定義”!
根據C++標准的規定,一個變量是聲明,必須同時滿足兩個條件,否則就是定義:
① 聲明必須使用extern關鍵字
① 聲明必須使用extern關鍵字
② 不能給變量賦初值
所以,下面的是聲明:
extern int a;
下面的是定義
int a;
下面的是定義
int a;
int a = 0;
extern int a =0;
【2】對於那么編程不是那么嚴謹的程序員,總是在需要使用變量的文件中隨意定義一個全局變量,並且對於變量名也不予考慮,這也往往容易造成變量名重復,而造成LNK2005錯誤。
(2)頭文件的包含重復。
往往需要包含的頭文件中含有變量、函數、類的定義,在其它使用的地方又不得不多次包含之,如果頭文件中沒有相關的宏等防止重復鏈接的措施,那么就會產生LNK2005錯誤。
【1】解決辦法是在需要包含的頭文件中做類似的處理:
#ifndef MY_H_FILE //如果沒有定義這個宏 #define MY_H_FILE //定義這個宏 ……. //頭文件主體內容 ……. #endif
【2】上面是使用宏來做的,也可以使用預編譯來做,在頭文件中加入
#pragma once //頭文件主體
【3】綜合兩者都 使用
#pragma once
#ifndef MY_H_FILE //如果沒有定義這個宏
#define MY_H_FILE //定義這個宏
……. //頭文件主體內容
…….
#endif
//頭文件主體
(3)
使用第三方的庫造成的。
這種情況主要是C運行期函數庫和MFC的庫沖突造成的。
具體的辦法就是
將那個提示出錯的庫放到另外一個庫的前面。
另外選擇不同的C函數庫,可能會引起這個錯誤。
微軟和C有兩種C運行期函數庫,【1】一種是普通的函數庫:LIBC.LIB,不支持多線程。【2】另外一種是支持多線程的:msvcrt.lib。
如果一個工程里,這兩種函數庫混合使用,可能會引起這個錯誤,一般情況下它需要MFC的庫先於C運行期函數庫被鏈接,因此建議使用支持多線程的msvcrt.lib。所以在使用第三方的庫之前首先要知道它鏈接的是什么庫,否則就可能造成LNK2005錯誤。
如果不得不使用第三方的庫,可以嘗試按 下面所說的方法修改,但不能保證一定能解決問題,前兩種方法是微軟提供的:
① 選擇VC菜單Project->Settings->Link->Catagory選擇Input,再在Ignore libraries 的Edit欄中填入你需要忽略的庫, 如:Nafxcwd.lib;Libcmtd.lib。然后在Object/library Modules的Edit欄中填入正確的庫的順序,這里需要你能確定什么是正確的順序,呵呵,God bless you!
② 選擇VC菜單Project->Settings->Link頁,然后在Project Options的Edit欄中輸入/verbose:lib,這樣就可以在編譯鏈接程序過程中在輸出窗口看到鏈接的順序了。
③ 選擇VC菜單Project->Settings->C/C++頁,Catagory選擇Code Generation后再在User Runtime libraray中選擇MultiThread DLL等其他庫,逐一嘗試。
參考網址:
http://www.cnblogs.com/MuyouSome/p/3332699.html
