基于C++類型重定義的使用詳解
更新時(shí)間:2013年05月16日 10:47:17 作者:
本篇文章是對C++中類型重定義的使用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
這幾天工作時(shí)碰到一個(gè)C++的編譯錯(cuò)誤(我使用的是Visual C++ 7.0),說是有一個(gè)類重復(fù)定義,仔細(xì)想想我們的這個(gè)項(xiàng)目也是做了好幾個(gè)Release了, 內(nèi)部代碼應(yīng)該不會(huì)有這樣的低級錯(cuò)誤, 真把類型給重復(fù)定義了,檢查結(jié)果正如我預(yù)料的一樣。 就這樣, 我左右沒找到原因,被一個(gè)編譯錯(cuò)誤給卡在那里了。(在我的概念中, 程序錯(cuò)誤的等級為:編譯錯(cuò)誤->鏈接錯(cuò)誤->邏輯錯(cuò)誤, 此錯(cuò)誤屬于最低級 )。這時(shí)我仔細(xì)看了一下錯(cuò)誤提示, 發(fā)現(xiàn)重復(fù)定義是由于從兩個(gè)不同的路徑包含了同一個(gè)頭文件而引起的,同事也建議從另外一個(gè)路徑打開工程試試, 這才慢慢發(fā)現(xiàn)了原因。這個(gè)原因可能有些拗口,而事實(shí)上要出現(xiàn)這種錯(cuò)誤也有些"曲折", 讓我從不同情況下的類型重定義來解釋一下吧。
我總結(jié)的類型重定義情況有三。
一、沒有在文件頭加#pragma once指示符。
Type1.h:
//#pragma once
class Type
{
};
Main.cpp:
#include "Type1.h"
#include "Type1.h"
int main(int argc, char *argv[])
{
return 1;
}
#pragma once的作用是保證本文件只被編譯一次,如果沒有在Type1.h中加這句話,那么在main.cpp里面包含了兩次Type1.h, 就相當(dāng)于在main.cpp里面定義了兩次Type類, 自然就是類型重定義了。
二、兩個(gè)不同的頭文件中定義了相同的類型(均有#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[])
{
return 1;
}
這里main.cpp中同時(shí)包含了Type1.h, Type2.h兩個(gè)頭文件, 雖然其文件頭都有#pragma once,但因?yàn)槭遣煌奈募?預(yù)編譯器還是會(huì)兩次把Type類的定義放在Main.cpp中, 所以也會(huì)出現(xiàn)了重定義。
三、從兩個(gè)不同的路徑包含了同一個(gè)頭文件
前面兩種是比較常見, 也是比較容易解決的情況, 而這里要講的第三種情況, 比較少見, 而且一般出現(xiàn)在有虛擬映射盤的時(shí)候。(這樣才能做到從兩個(gè)不同的路徑包含同一個(gè)頭文件), 其他會(huì)在什么時(shí)候出現(xiàn), 我還沒想到, 知道的朋友頂一下:)。下面我來分析一下:
1) 有VC工程在D:\Test目錄下。
2) 映射虛擬盤X為D:\Test.
不熟悉的網(wǎng)友可以按此操作: 開始->運(yùn)行->在運(yùn)行窗口輸入: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[])
{
return 1;
}
這里我們在main.cpp這樣包含了兩個(gè)頭文件, 從本質(zhì)上來講, 它們都對應(yīng)于物理盤D:\Test下的文件Type1.h, 是同一個(gè)文件。但在不同的操作下, VC對其有不同的解釋。#include "X:\Type1.h"用的是絕對路徑, 自然沒有什么異議, 但#include "Type1.h"卻有些變化:
•假如我從D:\Test\下打開工程, 那么#include "Type1.h"其實(shí)就是#include "D:\Test\Type1.h"
•假如從X:\下打開工程,那么#include "Type1.h"就解釋為#include "X:\Type1.h"
4) 在D:\Test下打開工程, 編譯, 出現(xiàn)類型Type重復(fù)定義錯(cuò)誤
這種情況下,main.cpp預(yù)編譯為:
Main.cpp:
#include "D:\Test\Type1.h"
#include "X:\Type1.h"
int main(int argc, char *argv[])
{
return 1;
}
#pragma once只保證本文件被編譯一次, 這里VC將其認(rèn)為是兩個(gè)不同的文件, 所以都要編譯, 出現(xiàn)編譯錯(cuò)誤自然也就不奇怪了。
當(dāng)然, 這里如果從X:\ 下打開工程的話,VC就會(huì)認(rèn)為都是從X:\Type1.h下包含這個(gè)文件,#pragma once起到了作用, 也就不會(huì)出現(xiàn)類型重定義了
四、總結(jié)
我在VC7, VC8,和Dev C++中都測試了第三種情況, 發(fā)現(xiàn)只有Dev C++是可以通過編譯的。這可能是微軟VC的#pragma once還不夠智能吧,輕易的被Windows的虛擬盤給蒙蔽了雙眼, 看不到其本質(zhì)(只是猜測, 或許VC這么處理是有其他用意的)。
因?yàn)樵谏源笠稽c(diǎn)的工程開發(fā)中, 我們一般都會(huì)用虛擬盤來方便工作, 一是訪問快捷,簡化了路徑, 二是因?yàn)槎嗳藚f(xié)同開發(fā),我們一般希望大家源代碼路徑相同,但我們不應(yīng)強(qiáng)制要求大家都把源代碼放死在某一目錄下, 這時(shí)把你放源代碼的路徑映射為一個(gè)虛擬盤(比如說統(tǒng)一為X:)就能把大家的代碼路徑統(tǒng)一起來了。但是另一方面,有了虛擬盤, 就為出現(xiàn)類型重定義提供了條件, 以下是我得出的兩個(gè)解決方法:
1) 拋棄#pragma once使用古老但集穩(wěn)定性與移植性于一身的
#ifndef _XXX_H
#define _XXX_H
//...#endif
來保證頭文件只被編譯一次。這樣不管是包含兩個(gè)相同的文件,還是包含兩個(gè)不同的文件,或是包含兩個(gè)文件相同但路徑不同的文件, 只要_XXX_H被定義過, 就不會(huì)再編譯那個(gè)編譯(但這里我們要保證_XXX_H的唯一性, 如果兩個(gè)不同的頭文件里用了同一_XXX_H,是會(huì)出問題的)
2) 在包含頭文件時(shí),不要使用絕對路徑, 哪怕那是虛擬盤的絕對路徑。
我總結(jié)的類型重定義情況有三。
一、沒有在文件頭加#pragma once指示符。
復(fù)制代碼 代碼如下:
Type1.h:
//#pragma once
class Type
{
};
Main.cpp:
#include "Type1.h"
#include "Type1.h"
int main(int argc, char *argv[])
{
return 1;
}
#pragma once的作用是保證本文件只被編譯一次,如果沒有在Type1.h中加這句話,那么在main.cpp里面包含了兩次Type1.h, 就相當(dāng)于在main.cpp里面定義了兩次Type類, 自然就是類型重定義了。
二、兩個(gè)不同的頭文件中定義了相同的類型(均有#pragma once)
復(fù)制代碼 代碼如下:
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[])
{
return 1;
}
這里main.cpp中同時(shí)包含了Type1.h, Type2.h兩個(gè)頭文件, 雖然其文件頭都有#pragma once,但因?yàn)槭遣煌奈募?預(yù)編譯器還是會(huì)兩次把Type類的定義放在Main.cpp中, 所以也會(huì)出現(xiàn)了重定義。
三、從兩個(gè)不同的路徑包含了同一個(gè)頭文件
前面兩種是比較常見, 也是比較容易解決的情況, 而這里要講的第三種情況, 比較少見, 而且一般出現(xiàn)在有虛擬映射盤的時(shí)候。(這樣才能做到從兩個(gè)不同的路徑包含同一個(gè)頭文件), 其他會(huì)在什么時(shí)候出現(xiàn), 我還沒想到, 知道的朋友頂一下:)。下面我來分析一下:
1) 有VC工程在D:\Test目錄下。
2) 映射虛擬盤X為D:\Test.
不熟悉的網(wǎng)友可以按此操作: 開始->運(yùn)行->在運(yùn)行窗口輸入:cmd->在cmd窗口輸入:
Subst X: D:\Test
3) 該工程有文件Type1.h, main.cpp
復(fù)制代碼 代碼如下:
Type1.h:
#pragma once
class Type{};
Main.cpp:
#include "Type1.h"
#include "X:\Type1.h"
int main(int argc, char *argv[])
{
return 1;
}
這里我們在main.cpp這樣包含了兩個(gè)頭文件, 從本質(zhì)上來講, 它們都對應(yīng)于物理盤D:\Test下的文件Type1.h, 是同一個(gè)文件。但在不同的操作下, VC對其有不同的解釋。#include "X:\Type1.h"用的是絕對路徑, 自然沒有什么異議, 但#include "Type1.h"卻有些變化:
•假如我從D:\Test\下打開工程, 那么#include "Type1.h"其實(shí)就是#include "D:\Test\Type1.h"
•假如從X:\下打開工程,那么#include "Type1.h"就解釋為#include "X:\Type1.h"
4) 在D:\Test下打開工程, 編譯, 出現(xiàn)類型Type重復(fù)定義錯(cuò)誤
這種情況下,main.cpp預(yù)編譯為:
復(fù)制代碼 代碼如下:
Main.cpp:
#include "D:\Test\Type1.h"
#include "X:\Type1.h"
int main(int argc, char *argv[])
{
return 1;
}
#pragma once只保證本文件被編譯一次, 這里VC將其認(rèn)為是兩個(gè)不同的文件, 所以都要編譯, 出現(xiàn)編譯錯(cuò)誤自然也就不奇怪了。
當(dāng)然, 這里如果從X:\ 下打開工程的話,VC就會(huì)認(rèn)為都是從X:\Type1.h下包含這個(gè)文件,#pragma once起到了作用, 也就不會(huì)出現(xiàn)類型重定義了
四、總結(jié)
我在VC7, VC8,和Dev C++中都測試了第三種情況, 發(fā)現(xiàn)只有Dev C++是可以通過編譯的。這可能是微軟VC的#pragma once還不夠智能吧,輕易的被Windows的虛擬盤給蒙蔽了雙眼, 看不到其本質(zhì)(只是猜測, 或許VC這么處理是有其他用意的)。
因?yàn)樵谏源笠稽c(diǎn)的工程開發(fā)中, 我們一般都會(huì)用虛擬盤來方便工作, 一是訪問快捷,簡化了路徑, 二是因?yàn)槎嗳藚f(xié)同開發(fā),我們一般希望大家源代碼路徑相同,但我們不應(yīng)強(qiáng)制要求大家都把源代碼放死在某一目錄下, 這時(shí)把你放源代碼的路徑映射為一個(gè)虛擬盤(比如說統(tǒng)一為X:)就能把大家的代碼路徑統(tǒng)一起來了。但是另一方面,有了虛擬盤, 就為出現(xiàn)類型重定義提供了條件, 以下是我得出的兩個(gè)解決方法:
1) 拋棄#pragma once使用古老但集穩(wěn)定性與移植性于一身的
復(fù)制代碼 代碼如下:
#ifndef _XXX_H
#define _XXX_H
//...#endif
來保證頭文件只被編譯一次。這樣不管是包含兩個(gè)相同的文件,還是包含兩個(gè)不同的文件,或是包含兩個(gè)文件相同但路徑不同的文件, 只要_XXX_H被定義過, 就不會(huì)再編譯那個(gè)編譯(但這里我們要保證_XXX_H的唯一性, 如果兩個(gè)不同的頭文件里用了同一_XXX_H,是會(huì)出問題的)
2) 在包含頭文件時(shí),不要使用絕對路徑, 哪怕那是虛擬盤的絕對路徑。
相關(guān)文章
C語言數(shù)據(jù)結(jié)構(gòu)通關(guān)時(shí)間復(fù)雜度和空間復(fù)雜度
對于一個(gè)算法,其時(shí)間復(fù)雜度和空間復(fù)雜度往往是相互影響的,當(dāng)追求一個(gè)較好的時(shí)間復(fù)雜度時(shí),可能會(huì)使空間復(fù)雜度的性能變差,即可能導(dǎo)致占用較多的存儲(chǔ)空間,這篇文章主要給大家介紹了關(guān)于C語言時(shí)間復(fù)雜度、空間復(fù)雜度的相關(guān)資料,需要的朋友可以參考下2022-04-04vscode調(diào)試使用make編譯的項(xiàng)目
VSCode本身是一個(gè)代碼編輯器,自帶的編譯功能比較弱,本文主要介紹了vscode調(diào)試使用make編譯的項(xiàng)目,具有一定的參考價(jià)值,感興趣的可以了解一下2023-10-10手動(dòng)添加bits/stdc++.h到vs2017的詳細(xì)步驟
這篇文章主要介紹了手動(dòng)添加bits/stdc++.h到vs2017的詳細(xì)步驟,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-02-02C++與QML進(jìn)行數(shù)據(jù)交互實(shí)現(xiàn)方式介紹
迫于無奈開始寫android的程序,以前使用QWidget的方式試過,雖然界面可以實(shí)現(xiàn),但是最后調(diào)用攝像頭時(shí),未能成功,再?zèng)]有繼續(xù)。這幾天開始使用qml進(jìn)行嘗試,在使用的過程中,其中的一個(gè)難點(diǎn),就是在qml與c++中數(shù)據(jù)的交互2022-09-09C++?反匯編之關(guān)于Switch語句的優(yōu)化措施
這篇文章主要介紹了C++?反匯編之關(guān)于Switch語句的優(yōu)化措施,利用三種優(yōu)化來降低樹高度,誰的效率高就優(yōu)先使用誰,三種優(yōu)化都無法匹配才會(huì)使用判定樹,具體內(nèi)容詳情跟隨小編一起看看吧2022-01-01