成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

從C#到C++容易出現的問題解答

開發 后端
本文從頭文件、namespace、繼承三個方面解答了從c#到c++的一些容易出現的困惑,希望會對大家有用。

前言

最近在學習用c++寫一下3D引擎(廢話一下,叫做WuguiEngine,首先Wugui是我的外號,也是代表這個引擎很粗糙,速度很慢,呵呵.之后等引擎成熟一點我再寫寫相關的一些文章).這幾天寫起來感覺c++好多地方和c#區別很大,這里大概寫寫這兩天碰到的一些問題,也許是從c#到c++的時候的一些通病,對c++ 新手也有一定的幫助.

另外在本文中,多數是將類拆分為.h文件和.cpp文件這樣對于工程來說更好管理.

另外閱讀本文需要一定的c++基礎,本文主要是一些雜碎的心得,而不是完整的教學,下面就開始解答從c#到c++的問題。

頭文件

頭文件是c#沒有的內容,所以用好頭文件是學習c++首先需要解決的一個內容.頭文件在正常的時候應該是只放上定義,實現代碼應該寫在同名的cpp文件里面

1) 重復包含:頭文件A.H被B.H與C.H同時包含,則在編譯的時候,A.H里面定義的代碼將被定義兩次,造成"Symbol already defined",在這種時候應該在.H文件頭尾處加上下列的代碼就可以解決這個問題:

  1. #ifndef _NAME_H  
  2.  #define _NAME_H  
  3.  //add defination  
  4. #endif 

其實#ifndef與#define的內容只要對于每個.h文件是唯一的就好了.

 2)嵌套包含:在A.H(class A)中使用了class B,又在B.H(class B)中使用了class A,這個時候需要在類的前面加入前置定義,下面的代碼:  

  1. #ifndef _A_H  
  2.    #define _A_H  
  3.       
  4.    #include "B.H"  
  5.       
  6.    //前置聲明  
  7.    class B;   
  8.       
  9.    class A  
  10.    {  
  11.       //add defination  
  12.    }  
  13.       
  14.    #endif 

這點說明,頭文件并不能代替前置聲明,對于有namespace的情況,前置聲明更加復雜,下一小節將談談這點內容.

3)一些經驗:

頭文件包含其實是一想很煩瑣的工作,不但我們看著累,編譯器編譯的時候也很累,再加上頭文件中常常出現的宏定義。感覺各種宏定義的展開是非常耗時間的,遠不如自定義函數來得速度。我僅就不同頭文件、源文件間的句則結構問題提出兩點原則,僅供參考:

第一個原則應該是,如果可以不包含頭文件,那就不要包含了。這時候前置聲明可以解決問題。如果使用的僅僅是一個類的指針,沒有使用這個類的具體對象(非指針),也沒有訪問到類的具體成員,那么前置聲明就可以了。因為指針這一數據類型的大小是特定的,編譯器可以獲知。

第二個原則應該是,盡量在CPP文件中包含頭文件,而非在頭文件中。假設類A的一個成員是是一個指向類B的指針,在類A的頭文件中使用了類B的前置聲明并便宜成功,那么在A的實現中我們需要訪問B的具體成員,因此需要包含頭文件,那么我們應該在類A的實現部分(CPP文件)包含類B的頭文件而非聲明部分(H文件)。

第三個原則是(我自己覺得的):盡量對引用的其他的類都加上前置聲明.這樣不僅使得程序的結構更加清楚,而且使得可以少出一些編譯錯誤

namespace

  1.  #ifndef _BASEGAME_H  
  2.  #define _BASEGAME_H  
  3.     
  4.  namespace WuguiEngine  
  5.  {  
  6.      //前置聲明  
  7.      class IUpdatable;  
  8.     
  9.      //前置聲明  
  10.     namespace Graphics  
  11.     {  
  12.         class GraphicsDevice;  
  13.     }  
  14.       
  15.     //前置聲明  
  16.     namespace Core  
  17.     {  
  18.         class TimeUsed;  
  19.     }  
  20.    
  21.     //簡化的寫法,GraphicsDevice可以在下面的代碼中直接使用  
  22.     //GraphicsDevice與WuguiEngine::Graphics::GraphicsDevice等價  
  23.     //注意:如果不加上下面這行,WuguiEngine::Graphics不能省略  
  24.     using Graphics::GraphicsDevice;  
  25.    
  26.     class BaseGame : virtual public IDisposed,   
  27.         virtual public IRenderable  
  28.     {  
  29.         // add definition  
  30.     }  
  31. }  
  32.    
  33. #endif 

上面的代碼簡單的說明了namespace的一些基本的用法,該段代碼取自BaseGame.H的文件,在class BaseGame中,用到了來自不同的命名空間下的IUpdatable, GraphicsDevice等類.具體的內容可以先看看代碼里面的注釋

在需要在BaseGame.H中包含這些類的頭文件的同時,需要加上這些類的前置聲明,如果不加會出現一些詭異的編譯錯誤(有些問題可能也是在于我比較不了解c++),反正我覺得使用using namespace ...并不是一個很好的做法,不如using XXX::YYY::ZZZ來得準確(這里也希望高手來拍磚指點一下.

繼承

1)virtual繼承

在繼承接口(只具有純虛函數的類)的時候最好為繼承的方法加上virtual的說明(參考上面一段的代碼,IUpdatable與IDisposed就是接口,主要的原因是在于c++的多重繼承機制,假如A與B都繼承自IInterface,C又繼承自A,B,那么普通的繼承方式則會在C類中保存兩個IInterface的副本.

而virtual繼承就有區別了,virutal繼承告訴編譯器:"我繼承的是一個純接口!",這樣只會保存一個副本.

2)多種繼承方式

大家知道,C++對于繼承的方式有很多的修飾,有public, private, protected,而且還可以加入virtual關鍵字.在c++中,public繼承與在c#中的繼承是沒有區別的,而其他幾種繼承是具有理論的價值,而實際中的應用得非常非常的少,這里就不再贅述了.

3)實驗:重寫虛函數

下面做一個簡單的實驗,看看c++與c#在重寫虛函數時候調用基類的虛函數的時候的情況,大家自己可以對比一下c#.測試代碼:

 

  1. #include < iostream>  
  2.  using namespace std;  
  3.      
  4.   class Base  
  5.   {  
  6.   public:  
  7.       virtual void Func() { cout < <  "Hello Earth" < <  endl; }  
  8.  };  
  9.      
  10.  class Derived : public Base  
  11.  {  
  12.  public:  
  13.      virtual void Func() { cout < <  "Hello World" < <  endl; }  
  14.  };  
  15.     
  16.  int main()  
  17.  {  
  18.      Base* p1 = new Base();  
  19.      p1->Func();  
  20.     
  21.      Base* p2 = new Derived();  
  22.      p2->Func();  
  23.  }  

輸出:

              Hello Earth

              Hello World

而在將Base的Func()改為純虛函數之后,在Base* p1 = new Base()這句話上出現編譯錯誤,提示不能實例化抽象類. 而對于p2的調用時正常的

下面繼續深入我們的實驗,三個類依次繼承,爺爺類具有一個純虛函數,爸爸類什么都不寫,兒子類重寫該函數,都使用virtual public繼承: 

  1. #include < iostream>  
  2.  using namespace std;  
  3.     
  4.  class Base  
  5.  {  
  6.  public:  
  7.      virtual void Func() = 0 ;  
  8.   };  
  9.      
  10.  class Derived : virtual public Base  
  11.  {  
  12.     
  13.  };  
  14.     
  15.  class DerivedDerived : virtual public Derived  
  16.  {  
  17.  public:  
  18.      virtual void Func() { cout < <  "Hello World" < <  endl; }  
  19.  };  
  20.     
  21.  int main()  
  22.  {  
  23.      Base* p1 = new DerivedDerived();  
  24.      Derived* p2 = new DerivedDerived();  
  25.     
  26.      p1->Func();  
  27.      p2->Func();  
  28.  }  

輸出結果都是"Hello World"

今天也有點晚了,明天還要坐火車,從c#到c++的問題先寫到這里,各位晚安,歡迎提出意見 :-D

【編輯推薦】

  1. C#事件和委托的編譯代碼
  2. C#事件的由來
  3. 學習C#委托:將方法綁定到委托
  4. 學習C#委托:將方法作為方法的參數
  5. C# 插件構架實例詳解
責任編輯:book05 來源: 博客園
相關推薦

2009-06-15 17:05:03

C#基元類型

2011-05-10 14:14:10

OSPF路由

2011-05-10 14:32:19

OSPF路由

2011-10-20 09:57:58

AS400 FTPFTP

2010-10-09 17:19:50

mysql存儲過程

2009-08-27 16:03:31

從c#到c++

2011-04-14 10:46:23

2009-12-25 10:59:08

WPF Timer

2009-10-13 14:56:00

CCNA培訓

2010-07-26 13:05:44

Perl子程序參數

2011-07-13 17:57:15

SQLite

2011-08-09 15:10:00

SQLite

2010-09-13 14:43:47

無線技術常見問題

2023-10-10 08:01:13

2009-10-27 11:10:56

linux問題解答

2009-12-03 18:09:51

Visual Stud

2011-07-21 11:19:51

JAVA

2020-10-22 07:22:41

物聯網網關物聯網IOT

2013-05-23 09:49:49

Chrome for google

2011-03-30 14:44:28

MRTG
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 看羞羞视频免费 | 97色在线视频 | 国产日产精品一区二区三区四区 | 亚洲人在线 | 本地毛片 | 在线亚洲免费视频 | 国产99久久精品 | 国产日韩欧美在线 | 免费一区二区三区 | 精品av| 国产精品日韩一区二区 | 亚洲午夜在线 | 精品一区二区三区四区在线 | 日韩一二区在线 | 国产精品69毛片高清亚洲 | 午夜视频在线免费观看 | 国产精品欧美一区二区三区不卡 | 中文字字幕一区二区三区四区五区 | 一区二区视频在线 | 成在线人视频免费视频 | 亚洲精品字幕 | 午夜精品视频在线观看 | 日韩精品一区中文字幕 | xxxxx黄色片| 搞av.com| 欧美精品成人一区二区三区四区 | 久草视频网站 | 国产一区 | 中文字幕一区二区三区四区五区 | 蜜桃五月天 | 91久久精品一区二区三区 | 美女视频一区二区三区 | 日本精品视频一区二区 | 日韩中文字幕第一页 | 国产精品欧美一区二区 | 国产欧美日韩 | 欧美激情精品久久久久久 | 久久精品国产一区二区电影 | 国产精品视频一二三区 | 欧美日韩精品国产 | 欧美精品video |