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

SmartPtr
本博客已搬至:http://www.cnblogs.com/baiyanhuang/
posts - 29,comments - 176,trackbacks - 0
By SmartPtr(http://m.shnenglu.com/SmartPtr/)

這幾天工作時碰到一個C++的編譯錯誤(我使用的是Visual C++ 7.0),說是有一個類重復定義,仔細想想我們的這個項目也是做了好幾個Release了, 內部代碼應該不會有這樣的低級錯誤, 真把類型給重復定義了,檢查結果正如我預料的一樣。 就這樣, 我左右沒找到原因,被一個編譯錯誤給卡在那里了。(在我的概念中, 程序錯誤的等級為:編譯錯誤->鏈接錯誤->邏輯錯誤, 此錯誤屬于最低級 )。這時我仔細看了一下錯誤提示, 發現重復定義是由于從兩個不同的路徑包含了同一個頭文件而引起的,同事也建議從另外一個路徑打開工程試試, 這才慢慢發現了原因。這個原因可能有些拗口,而事實上要出現這種錯誤也有些曲折 讓我從不同情況下的類型重定義來解釋一下吧。

我總結的C++中類型重定義情況有三。

1 沒有在文件頭加#pragma once指示符。

Type1.h:  #pragma once的作用是保證本文件只被編譯一次,如果沒有在Type1.h中加這句話那么在main.cpp里面包含了兩次Type1.h 就相當于在main.cpp里面定義了兩次Type類, 自然就是類型重定義了。

//#pragma once
class Type
{  

};

Main.cpp:

#include "Type1.h"
#include 
"Type1.h"
int main(int argc, char *argv[])
{
   
return 1;
}

 

2 兩個不同的頭文件中定義了相同的類型(均有#pragma once

Type1.h:Type2.h:Main.cpp:

#pragma once
class Type
{

};

 

#pragma once
class Type
{     

};

 

  這里main.cpp中同時包含了Type1.h, Type2.h兩個頭文件, 雖然其文件頭都有#pragma once,但因為是不同的文件, 預編譯器還是會兩次把Type類的定義放在Main.cpp中, 所以也會出現了重定義。

#include "Type1.h"
#include 
"Type2.h"
int main(int argc, char *argv[])
{
   
return 1;
}

 

3 從兩個不同的路徑包含了同一個頭文件

  前面兩種是比較常見, 也是比較容易解決的情況, 而這里要講的第三種情況, 比較少見, 而且一般出現在有虛擬映射盤的時候。(這樣才能做到從兩個不同的路徑包含同一個頭文件), 其他會在什么時候出現, 我還沒想到, 知道的朋友頂一下:)。下面我來分析一下:
1 VC工程在D:\Test目錄下。
2 映射虛擬盤XD:\Test.
不熟悉的網友可以按此操作: 開始->運行->在運行窗口輸入:cmd->cmd窗口輸入:
Subst X: D:\Test->回車。
3 該工程有文件Type1.h, main.cpp

Type1.h:

#pragma once
class Type
{

};

Main.cpp:

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"

#include "Type1.h"
#include 
"X:Type1.h"
int main(int argc, char *argv[])
{
   
return 1
;
}

    這里我們在

4 D:\Test下打開工程, 編譯, 出現類型Type重復定義錯誤

這種情況下,main.cpp預編譯為:

Main.cpp:

只保證本文件被編譯一次, 這里VC將其認為是兩個不同的文件, 所以都要編譯, 出現編譯錯誤自然也就不奇怪了。
   
當然, 這里如果從X:\ 下打開工程的話,VC就會認為都是從X:\Type1.h下包含這個文件,#pragma once起到了作用, 也就不會出現類型重定義了

#include "D:TestType1.h"
#include 
"X:Type1.h"
int main(int argc, char *argv[])
{
   
return 1;
}

  #pragma once

總結

我在VC7, VC8,Dev C++中都測試了第三種情況, 發現只有Dev C++是可以通過編譯的。這可能是微軟VC#pragma once還不夠智能吧,輕易的被Windows的虛擬盤給蒙蔽了雙眼, 看不到其本質(只是猜測, 或許VC這么處理是有其他用意的)。

因為在稍大一點的工程開發中, 我們一般都會用虛擬盤來方便工作, 一是訪問快捷,簡化了路徑, 二是因為多人協同開發,我們一般希望大家源代碼路徑相同,但我們不應強制要求大家都把源代碼放死在某一目錄下, 這時把你放源代碼的路徑映射為一個虛擬盤(比如說統一為X:)就能把大家的代碼路徑統一起來了。但是另一方面,有了虛擬盤, 就為出現類型重定義提供了條件, 以下是我得出的兩個解決方法:

1
拋棄#pragma once使用古老但集穩定性與移植性于一身的

來保證頭文件只被編譯一次。這樣不管是包含兩個相同的文件,還是包含兩個不同的文件,或是包兩個文件相同但路徑不同的文件, 只要_XXX_H被定義過, 就不會再編譯那個編譯(但這里我們要保證_XXX_H的唯一性, 如果兩個不同的頭文件里用了同一_XXX_H,是會出問題的)

#ifndef _XXX_H
#define _XXX_H
...

#endif

 

2 在包含頭文件時,不要使用絕對路徑, 哪怕那是虛擬盤的絕對路徑。

posted on 2007-07-05 17:49 SmartPtr 閱讀(5873) 評論(5)  編輯 收藏 引用

FeedBack:
# re: C++中的類型重定義
2007-07-05 23:08 | cpper
學到了!!!!!  回復  更多評論
  
# re: C++中的類型重定義
2007-07-06 01:52 | 至尊拖鞋
見識了,還沒用過虛擬盤映射功能的  回復  更多評論
  
# re: C++中的類型重定義
2007-07-06 14:32 | william
也不一定用subst來做虛擬盤,如果是ntfs格式的話 其實做一個虛擬目錄比較好  回復  更多評論
  
# re: C++中的類型重定義
2007-07-08 01:32 | 天津大學計算機學院 常興龍
總結得很不錯,在Symbian開發時會經常用的虛擬路徑。  回復  更多評論
  
# re: C++中的類型重定義
2007-07-12 15:35 | Rick

VC是用MS自己的Compiler,估計這個Compiler是采用頭文件的絕對路徑來作為唯一標識的,所以對于同一個文件通過不同的方式(本地和虛擬盤)來include,VC視作為兩個不同的頭文件。
而DevC++采用GCC的Compiler,這個Compiler發現兩個同名的頭文件(不管絕對路徑和這兩個文件是否其實是一個文件)被include后,會檢查這兩個頭文件的修改時間是否一致。如果一致,則認為這兩個頭文件是一樣的,并且頭文件包含“#pragma once”的話就只解析一次,不會報錯。否則,兩個文件修改時間不一樣,那會認為是兩個不同的文件(不管這兩個文件的內容是否一樣),所以解析的時候會報重復定義錯誤。
  回復  更多評論
  

只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲午夜黄色| 国产精品乱人伦一区二区 | 日韩亚洲精品视频| 亚洲第一黄色网| 久久亚洲精品伦理| 久久黄色影院| 久久婷婷久久| 毛片一区二区三区| 欧美激情小视频| 亚洲蜜桃精久久久久久久| 99视频一区二区| 亚洲欧美日韩中文视频| 久久国产手机看片| 欧美大片91| 国产精品老牛| 亚洲国产91精品在线观看| 亚洲精品视频二区| 亚洲免费影视| 久久综合狠狠综合久久综青草| 欧美chengren| 夜夜精品视频| 久久久99国产精品免费| 欧美国产欧美综合| 国产精品午夜视频| 亚洲国产一区二区三区青草影视 | 久久久999成人| 欧美国产视频在线| 亚洲图片欧美一区| 久久久久久久波多野高潮日日| 欧美激情偷拍| 亚洲欧洲另类国产综合| 老色批av在线精品| 一区二区三区蜜桃网| 久久久91精品国产| 国产精品播放| 亚洲精品久久久久久久久久久久久| 亚洲性线免费观看视频成熟| 久久夜色精品国产噜噜av| 日韩一区二区精品葵司在线| 久久精品视频亚洲| 国产精品视频免费观看| 国产精品毛片一区二区三区| 136国产福利精品导航| 亚洲综合999| 亚洲欧洲久久| 免费在线国产精品| 国产一区美女| 先锋影院在线亚洲| 99re66热这里只有精品4| 久久激情久久| 国产色产综合产在线视频| 亚洲一区二区不卡免费| 亚洲激精日韩激精欧美精品| 久久久久久国产精品mv| 国产曰批免费观看久久久| 性欧美8khd高清极品| 一区二区三区免费观看| 欧美三级视频在线观看| 日韩一级二级三级| 亚洲高清自拍| 欧美精品久久一区| 欧美日韩八区| 日韩一级在线观看| 亚洲国产精选| 欧美国产日韩一区二区三区| 亚洲黄色天堂| 欧美激情一二三区| 蜜桃久久精品乱码一区二区| 激情欧美国产欧美| 欧美va天堂| 欧美r片在线| 99精品欧美一区二区三区| 亚洲精品综合| 国产精品美女www爽爽爽视频| 亚洲一区尤物| 亚洲欧美日韩国产综合| 国产日韩综合一区二区性色av| 久久精品综合网| 久久一二三四| 夜夜嗨网站十八久久| 亚洲美女免费视频| 免费亚洲一区二区| 99国产精品| 亚洲一区二区视频在线| 国产深夜精品| 欧美国产日韩一二三区| 欧美日韩麻豆| 久久国产精品久久久久久久久久| 久久er99精品| 亚洲免费av网站| 欧美在线视频二区| 欧美一区二区三区喷汁尤物| 136国产福利精品导航网址| 亚洲精品1区2区| 国产女优一区| 亚洲高清资源| 国产日韩精品一区观看| 亚洲电影观看| 国产九九精品视频| 欧美高清视频www夜色资源网| 欧美日韩精品免费| 久久久久成人网| 欧美老女人xx| 快射av在线播放一区| 欧美日韩国产首页| 久久综合色天天久久综合图片| 欧美激情日韩| 久久野战av| 国产精品一区二区三区免费观看| 亚洲福利国产精品| 国产性猛交xxxx免费看久久| 亚洲精品一区二区三| 在线观看免费视频综合| 亚洲视频第一页| 日韩视频不卡中文| 久久久999国产| 欧美在线视频在线播放完整版免费观看 | 亚洲福利一区| 欧美一区二区在线播放| 亚洲在线视频网站| 欧美经典一区二区三区| 欧美成人黄色小视频| 国产视频一区在线观看一区免费| 亚洲精品欧美激情| 91久久久久久| 另类天堂av| 卡一卡二国产精品| 国产亚洲一区二区在线观看| 中文国产亚洲喷潮| 亚洲午夜精品久久久久久app| 欧美国产先锋| 亚洲精品欧美在线| 亚洲三级视频在线观看| 欧美成ee人免费视频| 亚洲国产裸拍裸体视频在线观看乱了中文 | 久久久久久久97| 欧美顶级艳妇交换群宴| 欧美高清视频一二三区| 国产欧美一区二区三区久久| 久久精品五月婷婷| 国产资源精品在线观看| 一区二区三区|亚洲午夜| 亚洲国产欧美精品| 欧美在线网站| 美玉足脚交一区二区三区图片| 国产精品久久久久9999吃药| 亚洲激情视频| 亚洲欧洲三级| 在线视频一区二区| 国产欧美日韩视频一区二区| 99精品欧美一区二区蜜桃免费| 亚洲激情成人网| 久久蜜臀精品av| 亚洲经典一区| 日韩午夜精品视频| 欧美不卡高清| 亚洲国产毛片完整版| 国产一区二区欧美| 久久中文在线| 欧美成人午夜激情在线| 亚洲成人在线网| 美女主播精品视频一二三四| 日韩视频在线免费观看| 日韩小视频在线观看专区| 久久综合色8888| 欧美成人69av| 亚洲人妖在线| 国产精品入口麻豆原神| 午夜精品福利视频| 久久久久久久综合狠狠综合| 国产视频在线观看一区| 欧美二区在线观看| 亚洲欧洲日本国产| 在线观看视频免费一区二区三区| 久久一区精品| 亚洲欧洲偷拍精品| 亚洲欧美久久久| 国产一区二区欧美| 日韩视频二区| 校园激情久久| 一区二区亚洲精品国产| 欧美在线亚洲| 亚洲午夜在线观看视频在线| 久久久久久久久久久一区| 在线日韩视频| 久久久久久尹人网香蕉| 亚洲图片激情小说| 久久精品人人做人人综合| 在线观看视频欧美| 国产精品s色| 欧美人牲a欧美精品| 亚洲欧美国产毛片在线| 美女啪啪无遮挡免费久久网站| 日韩亚洲欧美综合| 国产日韩欧美精品在线| 国产精品福利在线| 久久国产高清| 一本大道久久精品懂色aⅴ| 久久久精品网| 久久成人免费电影|