造成LNK2005錯(cuò)誤主要有以下幾種情況:
1.重復(fù)定義全局變量。可能存在兩種情況:
A、 對(duì)于一些初學(xué)編程的程序員,有時(shí)候會(huì)以為需要使用全局變量的地方就可以使用定義申明一下。其實(shí)這是錯(cuò)誤的,全局變量是針對(duì)整個(gè)工程的。正確的應(yīng)該是在一個(gè) CPP文件中定義如下:int g_Test;那么在使用的CPP文件中就應(yīng)該使用:extern int g_Test即可,如果還是使用int g_Test,那么就會(huì)產(chǎn)生LNK2005錯(cuò)誤,一般錯(cuò)誤錯(cuò)誤信息類似:AAA.obj error LNK2005 int book c? book@@3HA already defined in BBB.obj。切記的就是不能給變量賦值否則還是會(huì)有LNK2005錯(cuò)誤。
這里需要的是“聲明”,不是“定義”!根據(jù)C++標(biāo)準(zhǔn)的規(guī)定,一個(gè)變量是聲明,必須同時(shí)滿足兩個(gè)條件,否則就是定義:
(1)聲明必須使用extern關(guān)鍵字;(2)不能給變量賦初值
所以,下面的是聲明:
extern int a;
下面的是定義
int a; int a = 0; extern int a =0;
B、對(duì)于那么編程不是那么嚴(yán)謹(jǐn)?shù)某绦騿T,總是在需要使用變量的文件中隨意定義一個(gè)全局變量,并且對(duì)于變量名也不予考慮,這也往往容易造成變量名重復(fù),而造成LNK2005錯(cuò)誤。
2. 頭文件的包含重復(fù)。往往需要包含的頭文件中含有變量、函數(shù)、類的定義,在其它使用的地方又不得不多次包含之,如果頭文件中沒有相關(guān)的宏等防止重復(fù)鏈接的措 施,那么就會(huì)產(chǎn)生LNK2005錯(cuò)誤。解決辦法是在需要包含的頭文件中做類似的處理:#ifndef MY_H_FILE //如果沒有定義這個(gè)宏
#define MY_H_FILE //定義這個(gè)宏
……. //頭文件主體內(nèi)容
…….
#endif
上面是使用宏來做的,也可以使用預(yù)編譯來做,在頭文件中加入:
#pragma once
//頭文件主體
3. 使用第三方的庫造成的。這種情況主要是C運(yùn)行期函數(shù)庫和MFC的庫沖突造成的。具體的辦法就是將那個(gè)提示出錯(cuò)的庫放到另外一個(gè)庫的前面。另外選擇不同的C 函數(shù)庫,可能會(huì)引起這個(gè)錯(cuò)誤。微軟和C有兩種C運(yùn)行期函數(shù)庫,一種是普通的函數(shù)庫:LIBC.LIB,不支持多線程。另外一種是支持多線程的: msvcrt.lib。如果一個(gè)工程里,這兩種函數(shù)庫混合使用,可能會(huì)引起這個(gè)錯(cuò)誤,一般情況下它需要MFC的庫先于C運(yùn)行期函數(shù)庫被鏈接,因此建議使用 支持多線程的msvcrt.lib。所以在使用第三方的庫之前首先要知道它鏈接的是什么庫,否則就可能造成LNK2005錯(cuò)誤。如果不得不使用第三方的 庫,可以嘗試按下面所說的方法修改,但不能保證一定能解決問題,前兩種方法是微軟提供的:
A、選擇VC菜單Project-> Settings->Link->Catagory選擇Input,再在Ignore libraries 的Edit欄中填入你需要忽略的庫,如:Nafxcwd.lib;Libcmtd.lib。然后在Object/library Modules的Edit欄中填入正確的庫的順序,這里需要你能確定什么是正確的順序,呵呵,God bless you!
B、選擇VC菜單Project->Settings->Link頁,然后在Project Options的Edit欄中輸入/verbose:lib,這樣就可以在編譯鏈接程序過程中在輸出窗口看到鏈接的順序了。
C、選擇VC菜單Project->Settings->C/C++頁,Catagory選擇Code Generation后再在User Runtime libraray中選擇MultiThread DLL等其他庫,逐一嘗試。
這就是我所遇到過的LNK2005錯(cuò)誤的幾種情況,肯定還有其他的情況也可能造成這種錯(cuò)誤,所以我不希望你在看完這篇文章以后,再遇到LNK2005錯(cuò)誤時(shí)候,不動(dòng)腦筋的想對(duì)號(hào)入座的排除錯(cuò)誤。編程的過程就是一個(gè)思考的過程,所以還是多多開動(dòng)你的頭腦,那樣收獲會(huì)更多!
=======================================
支持,我在社區(qū)里也看到了許多LINK 2005錯(cuò)
補(bǔ)充一點(diǎn),就是一次在用第三方庫時(shí),由于errno被重定義,用多種方法都不能解決,后查找MSDN,發(fā)現(xiàn)link有個(gè)選項(xiàng)/FORCE可以解決,在IDE下
Project->Settings->Link頁,選categroy為custom,將force file output前打勾
但會(huì)有警告
warning LNK4088: image being generated due to /FORCE option; image may not run
但的確解決了問題,這是由于VC對(duì)重定義比較嚴(yán)格,像BCB或GCC在庫中的重定義不會(huì)有任何警告或錯(cuò)誤
========================================
我 發(fā)現(xiàn)的另外一個(gè)出現(xiàn)LINK2005的現(xiàn)象,好像是由于名稱空間而引起的。我在dos下寫的程序沒有問題,但是放在mfc中就出現(xiàn)了這個(gè)鏈接錯(cuò)誤。因?yàn)槠?初圖省事,我在一個(gè)頭文件中寫了using namespace std,并且這個(gè)頭文件我多處使用,另外,我還使用了boost庫。后來,問題解決的方法非常奇怪,在一個(gè)頭文件中引用其他頭文件,這些頭文件的順序換一 下就通過了,那個(gè)出現(xiàn)問題的頭文件中我使用了std::map,當(dāng)我把這種容器使用模板代替后,鏈接就有沒事了。(例如:template< class coll>),后來感到模板技術(shù)還有這種效果,賺了!哈哈
========================================
What are the C and C++ libraries my program would link with?
Summary Table
Compile Old New IOStream Libraries
Option IOStream or STL Linked With
|
|
/ML No No LIBC.LIB
|
/MLd No No LIBCD.LIB
|
/MT No No LIBCMT.LIB
|
/MTd No No LIBCMTD.LIB
|
/MD No No MSVCRT.LIB
|
/MDd No No MSVCRTD.LIB
|
/ML No Yes LIBC.LIB, LIBCP.LIB
|
/MLd No Yes LIBCD.LIB, LIBCPD.LIB
|
/MT No Yes LIBCMT.LIB, LIBCPMT.LIB
|
/MTd No Yes LIBCMTD.LIB, LIBCPMTD.LIB
|
/MD No Yes MSVCRT.LIB, MSVCPRT.LIB
|
/MDd No Yes MSVCRTD.LIB, MSVCPRTD.LIB
|
/ML Yes No LIBC.LIB, LIBCI.LIB
|
/MLd Yes No LIBCD.LIB, LIBCID.LIB
|
/MT Yes No LIBCMT.LIB, LIBCIMT.LIB
|
/MTd Yes No LIBCMTD.LIB, LIBCIMTD.LIB
|
/MD Yes No MSVCRT.LIB, MSVCIRT.LIB
|
/MDd Yes No MSVCRTD.LIB, MSVCIRTD.LIB
|
你的程序使用了/ML編譯選項(xiàng),而程序依賴的.lib可能使用/MDd選項(xiàng)編譯,造成鏈接沖突.
統(tǒng)一編譯選項(xiàng)可回避此錯(cuò)誤
Project Settings->C/C++ Tab->Category:CodeGeneration
Use run-time library組合框中選擇Multithread Dll(或Debug Multithread Dll )
Summary Table for CRT DLLs Used
Import Library Linked With |
DLLs Used (Visual C++ 5.0 |
從vc6升級(jí)到vc7的一些問題及解決方法- -
最近由于工作需要,把項(xiàng)目從vc6升級(jí)到vc7(vc.net2003)。升級(jí)過程遇到些問題,記錄于此。
1. 編譯時(shí)出現(xiàn):WINVER not defined. Defaulting to 0×0501 (Windows XP and Windows .NET Server)
這個(gè)問題是因?yàn)闆]有指定工程要使用的平臺(tái)SDK的版本。
Minimum system required Macros to define
Windows Server 2003 family _WIN32_WINNT>=0×0502
Windows XP _WIN32_WINNT>=0×0501
Windows 2000 _WIN32_WINNT>=0×0500
Windows NT 4.0 _WIN32_WINNT>=0×0400
Windows Me _WIN32_WINDOWS=0×0490
Windows 98 _WIN32_WINDOWS>=0×0410
Internet Explorer 6.0 _WIN32_IE>=0×0600
Internet Explorer 5.01, 5.5 _WIN32_IE>=0×0501
Internet Explorer 5.0, 5.0a, 5.0b _WIN32_IE>=0×0500
Internet Explorer 4.01 _WIN32_IE>=0×0401
Internet Explorer 4.0 _WIN32_IE>=0×0400
Internet Explorer 3.0, 3.01, 3.02 _WIN32_IE>=0×0300
解決辦法:
屬性,C/C++,命令行,附加項(xiàng)中添加 /D_WIN32_WINNT=0×0501 (因?yàn)槲沂窃趚p下工作的所以是0×0501)
2. Link是出現(xiàn):LINK : warning LNK4075: 忽略”/EDITANDCONTINUE”(由于”/INCREMENTAL:NO”規(guī)范)
這個(gè)問題是因?yàn)樵趘c6中,工程使用的增量編譯。
解決辦法:
屬性,鏈接器,常規(guī),啟動(dòng)增量鏈接 選擇 是(INCREMENTAL)
3. 編譯時(shí)出現(xiàn):warning C4129: “U” : 不可識(shí)別的字符轉(zhuǎn)義序列
error C3847: 通用字符中的錯(cuò)誤符號(hào);必須使用十六進(jìn)制數(shù)字
原 因:為開發(fā)全球通用的應(yīng)用程序,.NET Framework 使用 Unicode UTF-16(Unicode 轉(zhuǎn)換格式,16 位編碼形式)來表示字符。在某些情況下,.NET Framework 在內(nèi)部使用 UTF-8。引入通用字符名稱的格式是 u#### 或 U########。
解決辦法:
//#include MAKEPATH(MAIN_IMAGE_PATH, FunUtilUnit_star.txt)
#include “..ImageDataML128160FunUtilUnit_star.txt”
4. 鏈接是出現(xiàn):LIBCMTD.lib(crt0dat.obj) : error LNK2005: _exit 已經(jīng)在 MSVCRTD.lib(MSVCR71D.dll) 中定義 等類似錯(cuò)誤
原因:
Run-Time Library
?Run-Time Library是編譯器提供的標(biāo)準(zhǔn)庫,提供一些基本的庫函數(shù)和系統(tǒng)調(diào)用。
我們一般使用的Run-Time Library是C Run-Time Libraries。當(dāng)然也有Standard C++ libraries。
C Run-Time Libraries實(shí)現(xiàn)ANSI C的標(biāo)準(zhǔn)庫。VC安裝目錄的CRT目錄有C Run-Time庫的大部分源代碼。 C Run-Time Libraries有靜態(tài)庫版本,也有動(dòng)態(tài)鏈接庫版本;有單線程版本,也有多線程版本;還有調(diào)試和非調(diào)試版本。
?動(dòng)態(tài)鏈接庫版本:
/MD Multithreaded DLL 使用導(dǎo)入庫MSVCRT.LIB
/MDd Debug Multithreaded DLL 使用導(dǎo)入庫MSVCRTD.LIB
?靜態(tài)庫版本:
/ML Single-Threaded 使用靜態(tài)庫LIBC.LIB
/MLd Debug Single-Threaded 使用靜態(tài)庫LIBCD.LIB
/MT Multithreaded 使用靜態(tài)庫LIBCMT.LIB
/MTd Debug Multithreaded 使用靜態(tài)庫LIBCMTD.LIB
若要使用此運(yùn)行時(shí)庫 請忽略這些庫
單線程 (libc.lib) libcmt.lib、msvcrt.lib、libcd.lib、libcmtd.lib、msvcrtd.lib
多線程 (libcmt.lib) libc.lib、msvcrt.lib、libcd.lib、libcmtd.lib、msvcrtd.lib
使用 DLL 的多線程 (msvcrt.lib) libc.lib、libcmt.lib、libcd.lib、libcmtd.lib、msvcrtd.lib
調(diào)試單線程 (libcd.lib) libc.lib、libcmt.lib、msvcrt.lib、libcmtd.lib、msvcrtd.lib
調(diào)試多線程 (libcmtd.lib) libc.lib、libcmt.lib、msvcrt.lib、libcd.lib、msvcrtd.lib
使用 DLL 的調(diào)試多線程 (msvcrtd.lib) libc.lib、libcmt.lib、msvcrt.lib、libcd.lib、libcmtd.lib
解決方法:
屬性,鏈接器,輸入,忽略指定庫 libc.lib、libcmt.lib、msvcrt.lib、libcd.lib、libcmtd.lib (這是我需要忽略的,你可以根據(jù)你工程的實(shí)際情況選擇。)