這幾天工作時碰到乙個c++的編譯錯誤(我使用的是visual c++ 7.0),說是有乙個類重複定義,仔細想想我們的這個專案也是做了好幾個release了, 內部**應該不會有這樣的低階錯誤, 真把型別給重複定義了,檢查結果正如我預料的一樣。 就這樣, 我左右沒找到原因,被乙個編譯錯誤給卡在那裡了。(在我的概念中, 程式錯誤的等級為:編譯錯誤->鏈結錯誤->邏輯錯誤, 此錯誤屬於最低階 )。這時我仔細看了一下錯誤提示, 發現重複定義是由於從兩個不同的路徑包含了同乙個標頭檔案而引起的,同事也建議從另外乙個路徑開啟工程試試, 這才慢慢發現了原因。這個原因可能有些拗口,而事實上要出現這種錯誤也有些"曲折", 讓我從不同情況下的型別重定義來解釋一下吧。
我總結的型別重定義情況有三。
一、沒有在檔案頭加#pragma once指示符。
複製**
**如下:
type1.h:
//#pragma once
class type ;
main.cpp:
#include "type1.h"
#include "type1.h"
int main(int argc, char *argv)
#pragma once的作用是保證本檔案只被編譯一次,如果沒有在type1.h中加這句話,那麼在main.cpp裡面包含了兩次type1.h, 就相當於在main.cpp裡面定義了兩次type類, 自然就是型別重定義了。
二、兩個不同的標頭檔案中定義了相同的型別(均有#pragma once)
複製**
**如下:
type1.h:
#pragma once
class type ;
type2.h:
#pragma once
class type ;
main.cpp:
#include "type1.h"
#include "type2.h"
int main(int argc, char *argv)
這裡main.cpp中同時包含了type1.h, type2.h兩個標頭檔案, 雖然其檔案頭都有#pragma once,但因為是不同的檔案, 預編譯器還是會兩次把type類的定義放在main.cpp中, 所以也會出現了重定義。
三、從兩個不同的路徑包含了同乙個標頭檔案
前面兩種是比較常見, 也是比較容易解決的情況, 而這裡要講的第三種情況, 比較少見, 而且一般出現在有虛擬對映盤的時候。(這樣才能做到從兩個不同的路徑包含同乙個標頭檔案), 其他會在什麼時候出現, 我還沒想到, 知道的朋友頂一下:)。下面我來分析一下:
1)有vc工程在d:\test目錄下。
2)對映虛擬盤x為d:\test.
不熟悉的網友可以按此操作: 開始->執行->在執行視窗輸入:cmd->在cmd視窗輸入:
subst x: d:\test
3)該工程有檔案type1.h, main.cpp
複製**
**如下:
type1.h:
#pragma once
class type{};
main.cpp:
#include "type1.h"
#include "x:\type1.h"
int main(int argc, char *argv)
這裡我們在main.cpp這樣包含了兩個標頭檔案, 從本質上來講, 它們都對應於物理盤d:\test下的檔案type1.h, 是同乙個檔案。但在不同的操作下, vc對其有不同的解釋。#include "x:\type1.h"用的是絕對路徑, 自然沒有什麼異議, 但#include "type1.h"卻有些變化:
•假如我從d:\test\下開啟工程, 那麼#include "type1.h"其實就是#include "d:\test\type1.h"
•假如從x:\下開啟工程,那麼#include "type1.h"就解釋為#include "x:\type1.h"
4)在d:\test下開啟工程, 編譯, 出現型別type重複定義錯誤
這種情況下,main.cpp預編譯為:
複製**
**如下:
main.cpp:
#include "d:\test\type1.h"
#include "x:\type1.h"
int main(int argc, char *argv)
#pragma once只保證本檔案被編譯一次, 這裡vc將其認為是兩個不同的檔案, 所以都要編譯, 出現編譯錯誤自然也就不奇怪了。
當然, 這裡如果從x:\ 下開啟工程的話,vc就會認為都是從x:\type1.h下包含這個檔案,#pragma once起到了作用, 也就不會出現型別重定義了
四、總結
我在vc7, vc8,和dev c++中都測試了第三種情況, 發現只有dev c++是可以通過編譯的。這可能是微軟vc的#pragma once還不夠智慧型吧,輕易的被windows的虛擬盤給蒙蔽了雙眼, 看不到其本質(只是猜測, 或許vc這麼處理是有其他用意的)。
因為在稍大一點的工程開發中, 我們一般都會用虛擬盤來方便工作, 一是訪問快捷,簡化了路徑, 二是因為多人協同開發,我們一般希望大家源**路徑相同,但我們不應強制要求大家都把源**放死在某一目錄下, 這時把你放源**的路徑對映為乙個虛擬盤(比如說統一為x:)就能把大家的**路徑統一起來了。但是另一方面,有了虛擬盤, 就為出現型別重定義提供了條件, 以下是我得出的兩個解決方法:
1) 拋棄#pragma once使用古老但集穩定性與移植性於一身的
複製**
**如下:
#ifndef _***_h
#define _***_h
//...#endif
來保證標頭檔案只被編譯一次。這樣不管是包含兩個相同的檔案,還是包含兩個不同的檔案,或是包含兩個檔案相同但路徑不同的檔案, 只要_***_h被定義過, 就不會再編譯那個編譯(但這裡我們要保證_***_h的唯一性, 如果兩個不同的標頭檔案裡用了同一_***_h,是會出問題的)
2) 在包含標頭檔案時,不要使用絕對路徑, 哪怕那是虛擬盤的絕對路徑。
C 中的型別重定義
發現重複定義是由於從兩個不同的路徑包含了同乙個標頭檔案而引起的,同事也建議從另外乙個路徑開啟工程試試,這才慢慢發現了原因。這個原因可能有些拗口,而事實上要出現這種錯誤也有些 曲折 讓我從不同情況下的型別重定義來解釋一下吧。我總結的型別重定義情況有三。一 沒有在檔案頭加 pragma once指示符。...
struct型別重定義 C 學習大綱 結構型別
c 結構型別 結構型別用於表示由固定多個 型別可以不同的元素所構成的復合資料型別。1.結構型別定義 struct 或 typedef struct 1 別名可以跟結構型別名不一樣,但是一般都是一樣的,設定別名是為了方便像其他變數型別一樣定義變數,這是保留了c的語法。2 在結構型別定義時,對成員變數進...
C語言 自定義型別詳解
目錄 直接上 struct point p1 建立結構體時順便建立變數,分號一定不能掉 struct point p2 單獨建立變數 struct point p3 建立變數時順便賦值 struct node n1 int main struct是建立結構體的關鍵字,point是結構體的名字,p1為...