亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

C++中的類型轉換static_cast、dynamic_cast、const_cast和reinterpret_cast總結

2020-05-23 14:20:58
字體:
來源:轉載
供稿:網友
這篇文章主要介紹了C++中的類型轉換static_cast、dynamic_cast、const_cast和reinterpret_cast總結,需要的朋友可以參考下
 
 

前言

這篇文章總結的是C++中的類型轉換,這些小的知識點,有的時候,自己不是很注意,但是在實際開發中確實經常使用的。俗話說的好,不懂自己寫的代碼的程序員,不是好的程序員;如果一個程序員對于自己寫的代碼都不懂,只是知道一昧的的去使用,終有一天,你會迷失你自己的。

C++中的類型轉換分為兩種:

1.隱式類型轉換;
2.顯式類型轉換。

而對于隱式變換,就是標準的轉換,在很多時候,不經意間就發生了,比如int類型和float類型相加時,int類型就會被隱式的轉換位float類型,然后再進行相加運算。而關于隱式轉換不是今天總結的重點,重點是顯式轉換。在標準C++中有四個類型轉換符:static_cast、dynamic_cast、const_cast和reinterpret_cast;下面將對它們一一的進行總結。

static_cast

static_cast的轉換格式:static_cast <type-id> (expression)

將expression轉換為type-id類型,主要用于非多態類型之間的轉換,不提供運行時的檢查來確保轉換的安全性。主要在以下幾種場合中使用:

1.用于類層次結構中,基類和子類之間指針和引用的轉換;
當進行上行轉換,也就是把子類的指針或引用轉換成父類表示,這種轉換是安全的;
當進行下行轉換,也就是把父類的指針或引用轉換成子類表示,這種轉換是不安全的,也需要程序員來保證;

2.用于基本數據類型之間的轉換,如把int轉換成char,把int轉換成enum等等,這種轉換的安全性需要程序員來保證;

3.把void指針轉換成目標類型的指針,是及其不安全的;

注:static_cast不能轉換掉expression的const、volatile和__unaligned屬性。

dynamic_cast

dynamic_cast的轉換格式:dynamic_cast <type-id> (expression)

將expression轉換為type-id類型,type-id必須是類的指針、類的引用或者是void *;如果type-id是指針類型,那么expression也必須是一個指針;如果type-id是一個引用,那么expression也必須是一個引用。

dynamic_cast主要用于類層次間的上行轉換和下行轉換,還可以用于類之間的交叉轉換。在類層次間進行上行轉換時,dynamic_cast和static_cast的效果是一樣的;在進行下行轉換時,dynamic_cast具有類型檢查的功能,比static_cast更安全。在多態類型之間的轉換主要使用dynamic_cast,因為類型提供了運行時信息。下面我將分別在以下的幾種場合下進行dynamic_cast的使用總結:

1.最簡單的上行轉換

比如B繼承自A,B轉換為A,進行上行轉換時,是安全的,如下:

復制代碼代碼如下:

#include <iostream>
using namespace std;
class A
{
     // ......
};
class B : public A
{
     // ......
};
int main()
{
     B *pB = new B;
     A *pA = dynamic_cast<A *>(pB); // Safe and will succeed
}

 

2.多重繼承之間的上行轉換

C繼承自B,B繼承自A,這種多重繼承的關系;但是,關系很明確,使用dynamic_cast進行轉換時,也是很簡單的:

 

復制代碼代碼如下:

class A
{
     // ......
};
class B : public A
{
     // ......
};
class C : public B
{
     // ......
};
int main()
{
     C *pC = new C;
     B *pB = dynamic_cast<B *>(pC); // OK
     A *pA = dynamic_cast<A *>(pC); // OK
}

 

而上述的轉換,static_cast和dynamic_cast具有同樣的效果。而這種上行轉換,也被稱為隱式轉換;比如我們在定義變量時經常這么寫:B *pB = new C;這和上面是一個道理的,只是多加了一個dynamic_cast轉換符而已。

3.轉換成void *

可以將類轉換成void *,例如:

 

復制代碼代碼如下:

class A
{
public:
     virtual void f(){}
     // ......
};
class B
{
public:
     virtual void f(){}
     // ......
};
int main()
{
     A *pA = new A;
     B *pB = new B;
     void *pV = dynamic_cast<void *>(pA); // pV points to an object of A
     pV = dynamic_cast<void *>(pB); // pV points to an object of B
}

 

但是,在類A和類B中必須包含虛函數,為什么呢?因為類中存在虛函數,就說明它有想讓基類指針或引用指向派生類對象的情況,此時轉換才有意義;由于運行時類型檢查需要運行時類型信息,而這個信息存儲在類的虛函數表中,只有定義了虛函數的類才有虛函數表。

4.如果expression是type-id的基類,使用dynamic_cast進行轉換時,在運行時就會檢查expression是否真正的指向一個type-id類型的對象,如果是,則能進行正確的轉換,獲得對應的值;否則返回NULL,如果是引用,則在運行時就會拋出異常;例如:

 

復制代碼代碼如下:

class B
{
     virtual void f(){};
};
class D : public B
{
     virtual void f(){};
};
void main()
{
     B* pb = new D;   // unclear but ok
     B* pb2 = new B;
     D* pd = dynamic_cast<D*>(pb);   // ok: pb actually points to a D
     D* pd2 = dynamic_cast<D*>(pb2);   // pb2 points to a B not a D, now pd2 is NULL
}

 

這個就是下行轉換,從基類指針轉換到派生類指針。
對于一些復雜的繼承關系來說,使用dynamic_cast進行轉換是存在一些陷阱的;比如,有如下的一個結構:

C++中的類型轉換static_cast、dynamic_cast、const_cast和reinterpret_cast總結

D類型可以安全的轉換成B和C類型,但是D類型要是直接轉換成A類型呢?

 

復制代碼代碼如下:

class A
{
     virtual void Func() = 0;
};
class B : public A
{
     void Func(){};
};
class C : public A
{
     void Func(){};
};
class D : public B, public C
{
     void Func(){}
};
int main()
{
     D *pD = new D;
     A *pA = dynamic_cast<A *>(pD); // You will get a pA which is NULL
}

 

如果進行上面的直接轉,你將會得到一個NULL的pA指針;這是因為,B和C都繼承了A,并且都實現了虛函數Func,導致在進行轉換時,無法進行抉擇應該向哪個A進行轉換。正確的做法是:

復制代碼代碼如下:

int main()
{
     D *pD = new D;
     B *pB = dynamic_cast<B *>(pD);
     A *pA = dynamic_cast<A *>(pB);
}

 

這就是我在實現QueryInterface時,得到IUnknown的指針時,使用的是*ppv = static_cast<IX *>(this);而不是*ppv = static_cast<IUnknown *>(this);

對于多重繼承的情況,從派生類往父類的父類進行轉時,需要特別注意;比如有下面這種情況:

C++中的類型轉換static_cast、dynamic_cast、const_cast和reinterpret_cast總結

現在,你擁有一個A類型的指針,它指向E實例,如何獲得B類型的指針,指向E實例呢?如果直接進行轉的話,就會出現編譯器出現分歧,不知道是走E->C->B,還是走E->D->B。對于這種情況,我們就必須先將A類型的指針進行下行轉換,獲得E類型的指針,然后,在指定一條正確的路線進行上行轉換。

上面就是對于dynamic_cast轉換的一些細節知識點,特別是對于多重繼承的情況,在實際項目中,很容易出現問題。

const_cast

const_cast的轉換格式:const_cast <type-id> (expression)

const_cast用來將類型的const、volatile和__unaligned屬性移除。常量指針被轉換成非常量指針,并且仍然指向原來的對象;常量引用被轉換成非常量引用,并且仍然引用原來的對象??匆韵碌拇a例子:

復制代碼代碼如下:

/*
** FileName     : ConstCastDemo
** Author       : Jelly Young
** Date         : 2013/12/27
** Description  : More information, please go to http://www.49028c.com
*/
#include <iostream>
using namespace std;
class CA
{
public:
     CA():m_iA(10){}
     int m_iA;
};
int main()
{
     const CA *pA = new CA;
     // pA->m_iA = 100; // Error
     CA *pB = const_cast<CA *>(pA);
     pB->m_iA = 100;
     // Now the pA and the pB points to the same object
     cout<<pA->m_iA<<endl;
     cout<<pB->m_iA<<endl;
     const CA &a = *pA;
     // a.m_iA = 200; // Error
     CA &b = const_cast<CA &>(a);
     pB->m_iA = 200;
     // Now the a and the b reference to the same object
     cout<<b.m_iA<<endl;
     cout<<a.m_iA<<endl;
}

 

注:你不能直接對非指針和非引用的變量使用const_cast操作符去直接移除它的const、volatile和__unaligned屬性。

reinterpret_cast

reinterpret_cast的轉換格式:reinterpret_cast <type-id> (expression)

允許將任何指針類型轉換為其它的指針類型;聽起來很強大,但是也很不靠譜。它主要用于將一種數據類型從一種類型轉換為另一種類型。它可以將一個指針轉換成一個整數,也可以將一個整數轉換成一個指針,在實際開發中,先把一個指針轉換成一個整數,在把該整數轉換成原類型的指針,還可以得到原來的指針值;特別是開辟了系統全局的內存空間,需要在多個應用程序之間使用時,需要彼此共享,傳遞這個內存空間的指針時,就可以將指針轉換成整數值,得到以后,再將整數值轉換成指針,進行對應的操作。

總結

這篇博文總結了C++中的類型轉換,重點總結了其中的顯式轉換。對于C++支持的這四種顯式轉換都進行了詳細的描述。如果大家有什么補充的,或者我總結的有誤的地方,請大家多多指教。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
91精品久久久久久久久久久| 中文字幕日韩高清| 久久全球大尺度高清视频| 日韩国产精品视频| 欧美一级视频一区二区| 久久成人综合视频| 91精品国产精品| 日本成人在线视频网址| 在线观看亚洲区| 欧美日韩一区二区在线播放| 最近2019中文字幕一页二页| 在线观看精品国产视频| 久久躁狠狠躁夜夜爽| 国产精品丝袜高跟| 久久久亚洲国产| 69国产精品成人在线播放| 日韩精品中文字幕在线观看| 亚洲激情自拍图| 久久亚洲精品网站| 欧美xxxwww| 中文字幕不卡在线视频极品| 亚洲va电影大全| 一本一本久久a久久精品牛牛影视| 日韩精品免费综合视频在线播放| 亚洲福利小视频| 久久久久久亚洲精品中文字幕| 亚洲天堂av在线免费| 亚洲视频精品在线| 欧美高清理论片| 久久97精品久久久久久久不卡| 国产一区在线播放| 91美女片黄在线观看游戏| 亚洲欧洲视频在线| 狠狠色噜噜狠狠狠狠97| 日韩av在线免费观看| 国产91精品在线播放| 在线日韩第一页| 亚洲天堂av综合网| 欧美日韩国产成人高清视频| 中文字幕无线精品亚洲乱码一区| 欧美日韩成人精品| 国产精品久久二区| 欧美大片第1页| 欧美国产日韩中文字幕在线| 日韩av手机在线看| 国产在线精品成人一区二区三区| 国产在线视频不卡| 亚洲精品美女在线| 欧美在线激情视频| 久久久久国产一区二区三区| 亚洲午夜国产成人av电影男同| 久久在线观看视频| 精品成人av一区| 日韩av最新在线观看| 国产热re99久久6国产精品| 91国产美女在线观看| 疯狂做受xxxx高潮欧美日本| 中文字幕亚洲无线码在线一区| 欧美性猛交99久久久久99按摩| 亚洲高清不卡av| 欧美裸体xxxxx| 日韩高清电影免费观看完整版| 亚洲精品乱码久久久久久金桔影视| 亚洲欧美日韩爽爽影院| 久久不射热爱视频精品| 亚洲午夜精品视频| 久久精品2019中文字幕| 国产极品jizzhd欧美| 一本一本久久a久久精品牛牛影视| 久久久久一本一区二区青青蜜月| 九色精品美女在线| 国产精品香蕉av| 疯狂蹂躏欧美一区二区精品| 成人中文字幕+乱码+中文字幕| 国产精品三级久久久久久电影| 美女精品视频一区| 国产一区二区日韩| 国产欧美日韩综合精品| 岛国av一区二区| 国产97在线视频| 亚洲国产99精品国自产| 欧美怡春院一区二区三区| 亚洲国产另类 国产精品国产免费| 69国产精品成人在线播放| 国产精品美女午夜av| 国产精品成人av性教育| 久久久国产精品免费| 欧美丝袜一区二区三区| 亚洲成人av片| 国产精品亚洲视频在线观看| 亚洲最大在线视频| 亚洲精品视频播放| 国产综合福利在线| 国产一区二区三区网站| 色综合伊人色综合网| 国产精品成人观看视频国产奇米| 国产精品三级美女白浆呻吟| 国产成人精品在线播放| 97国产精品人人爽人人做| 国产成人拍精品视频午夜网站| 欧美高清视频在线播放| 不卡在线观看电视剧完整版| 精品国产一区二区三区久久久| 亚洲成人av片| 日韩免费在线播放| 色www亚洲国产张柏芝| 亚洲精品久久久久久久久| 中文字幕视频一区二区在线有码| 国产精品色婷婷视频| 亚洲人成网站免费播放| 亚洲国产一区二区三区在线观看| 国产成人午夜视频网址| 欧美日韩电影在线观看| 中文字幕少妇一区二区三区| 日本中文字幕成人| 91视频-88av| 国外成人性视频| 国产精品羞羞答答| 91亚洲精品久久久| 亚洲精品国产美女| 久久久久久成人精品| 亚洲一区二区久久久| 日韩在线欧美在线| 国产精品美女久久久久av超清| 亚洲欧洲偷拍精品| 亚洲久久久久久久久久| 国产亚洲a∨片在线观看| 日韩精品视频免费在线观看| 亚洲美女自拍视频| 久久噜噜噜精品国产亚洲综合| 亚洲人成亚洲人成在线观看| 日韩av手机在线看| 国产在线视频一区| 一区二区三区视频免费| 欧美激情精品久久久久久久变态| 亚洲欧美制服综合另类| 亚洲理论在线a中文字幕| 中文字幕精品网| 亚洲美女激情视频| 自拍偷拍亚洲区| 国产精品中文字幕久久久| 黑人巨大精品欧美一区二区免费| 青青久久av北条麻妃海外网| 亚洲成人久久久| 国产精品美女在线| 伊人伊成久久人综合网小说| 亚洲三级av在线| 亚洲成人性视频| 午夜精品一区二区三区在线播放| 亚洲综合社区网| 在线观看中文字幕亚洲| 深夜精品寂寞黄网站在线观看| 国产91在线播放九色快色| 国产性猛交xxxx免费看久久| 亚洲视频精品在线| 欧美日韩国产成人高清视频| 中文字幕日韩免费视频| 国产精品午夜国产小视频| 91亚洲一区精品| 国模视频一区二区三区| 亚洲白虎美女被爆操| 久久免费福利视频| 亚洲xxxx18| 日韩经典中文字幕|