關(guān)于C++的強(qiáng)制類型轉(zhuǎn)換淺析
前言
一說起強(qiáng)制類型轉(zhuǎn)換大家都很熟悉,相信很多學(xué)習(xí)完C++的朋友還在使用C語言的強(qiáng)制類型的方式 (類型)變量.
C++其實(shí)也具有自己的一套強(qiáng)制類型轉(zhuǎn)換它們分明是:static_cast reinterpret_cast const_cast dynamic_cast四種類型.
那么肯定會(huì)有人好奇C++是不是閑,C語言的強(qiáng)制類型用的舒舒服服的,為什么要新推出來這幾個(gè)?
新類型的強(qiáng)制轉(zhuǎn)換可以提供更好的控制強(qiáng)制轉(zhuǎn)換過程,允許控制各種不同種類的強(qiáng)制轉(zhuǎn)換。C++中風(fēng)格是static_cast<type>。C++風(fēng)格的強(qiáng)制轉(zhuǎn)換其他的好處是,它們能更清晰的表明它們要干什么。程序員只要掃一眼這樣的代碼,就能立即知道一個(gè)強(qiáng)制轉(zhuǎn)換的目
的。
static_cast
static_cast用于非多態(tài)類型的轉(zhuǎn)換(靜態(tài)轉(zhuǎn)換),任何標(biāo)準(zhǔn)轉(zhuǎn)換都可以用它,但它不能用于兩個(gè)不相關(guān)的類型進(jìn)行轉(zhuǎn)換.
何為不相關(guān)類型? 比如int 和 double char short就是相關(guān)類型. 和int*就是不相關(guān)類型.
我們來看一看static_cast的用法. 例如,通過將一個(gè)運(yùn)算對象強(qiáng)制轉(zhuǎn)換成double類型就能使表達(dá)式執(zhí)行浮點(diǎn)數(shù)除法:
double slope = static_cast<double>(j) / i;
當(dāng)static_cast需要把一個(gè)較大的算術(shù)類型賦值給較小的類型時(shí),static_cast非常有用。此時(shí),強(qiáng)制類型轉(zhuǎn)換告訴程序的讀者和編譯器:我們知道并且不在乎潛在的精度損失。一般來說,如果編譯器發(fā)現(xiàn)一個(gè)的算術(shù)類型試圖賦值給較小的類型,就會(huì)給出警告信息;但是當(dāng)我們執(zhí)行了顯式的類型轉(zhuǎn)換后,警告信息就會(huì)被關(guān)閉了。
reinterpret_cast
reinterpret_cast有著和C風(fēng)格的強(qiáng)制轉(zhuǎn)換同樣的能力。它可以轉(zhuǎn)化任何內(nèi)置的數(shù)據(jù)類型為其他任何的數(shù)據(jù)類型,也可以轉(zhuǎn)化任何指針類型為其他的類型。它甚至可以轉(zhuǎn)化內(nèi)置的數(shù)據(jù)類型為指針,無須考慮類型安全或者常量的情形。不到萬不得已絕對用。
因?yàn)閞einterpret_cast是一個(gè)蠻bug的操作,下面我來演示一下.
typedef void (* FUNC)(); int DoSomething (int i) { cout<<"DoSomething" <<endl; return 0; } void Test () { // reinterpret_cast可以編譯器以FUNC的定義方式去看待 DoSomething函數(shù) // 所以非常的BUG,下面轉(zhuǎn)換函數(shù)指針的代碼是不可移植的,所以不建議這樣用 // C++不保證所有的函數(shù)指針都被一樣的使用,所以這樣用有時(shí)會(huì)產(chǎn)生不確定的結(jié)果 FUNC f = reinterpret_cast< FUNC>(DoSomething ); f(); }
當(dāng)你這樣運(yùn)行的時(shí)候,你會(huì)發(fā)現(xiàn)通過函數(shù)指針沒有傳參數(shù)調(diào)用這個(gè)有參數(shù)的函數(shù)居然可以調(diào)用,這就很尷尬了,所以我告訴你不到萬不得已就不要使用reinterpret_cast
const_cast
對于將常量對象轉(zhuǎn)化為非常量對象的行為,我們一般稱之為“去掉const性質(zhì)”。一旦我們?nèi)サ袅四硞€(gè)對象的const性質(zhì),編譯器就不再阻止我們對該對象進(jìn)行寫操作了。如果對象本身不是一個(gè)常量,使用強(qiáng)制類型轉(zhuǎn)換獲得寫權(quán)限是合法的行為。然而如果對象是一個(gè)常量,再使用const_cast執(zhí)行寫操作就會(huì)產(chǎn)生未定義的后果。
舉個(gè)例子:
#include<iostream> #include<Windows.h> #include<assert.h> using namespace std; int main() { const int a = 2; int *p = const_cast<int*>(&a); *p = 3; cout << a << endl; system("pause"); return 0; }
我們有理由的認(rèn)為在內(nèi)存當(dāng)中a的值被修改為3,但是結(jié)果呢? 我們來看一看
這不科學(xué)???? 我們再打開監(jiān)視窗口看一下a的值.
我們都知道監(jiān)視窗口看到的都是從內(nèi)存當(dāng)中拿到的,但是為什么內(nèi)存當(dāng)中為3,打印出來就是2呢? 我來解釋一下.
C++編譯器具有優(yōu)化功能,當(dāng)你定一個(gè)const的常量的時(shí)候,系統(tǒng)覺得它不會(huì)被改變了,于是做一個(gè)優(yōu)化把該常量存到寄存器當(dāng)中,下次訪問的過程更快速一點(diǎn). 所以當(dāng)顯示窗口讀取數(shù)據(jù)的時(shí)候,他會(huì)直接去寄存器當(dāng)中讀取數(shù)據(jù).而不是去內(nèi)存,所以導(dǎo)致我們明明該掉了a的值,卻打印不出來,但是如何解決這個(gè)問題呢??
c++有一個(gè)關(guān)鍵字: volatile 該關(guān)鍵字的作用防止編譯器優(yōu)化,這個(gè)時(shí)候要輸出a就會(huì)老老實(shí)實(shí)的回內(nèi)存去查看.
#include<iostream> #include<Windows.h> #include<assert.h> using namespace std; int main() { volatile const int a = 2; int *p = const_cast<int*>(&a); *p = 3; cout << a << endl; system("pause"); return 0; }
dynamic_cast
前三種的強(qiáng)制類型轉(zhuǎn)換,他們能做到的C語言的強(qiáng)制類型轉(zhuǎn)換也大多能做到,最后一種C語言的強(qiáng)制類型轉(zhuǎn)換就沒有辦法了.
在類的轉(zhuǎn)換時(shí),在類層次間進(jìn)行上行轉(zhuǎn)換時(shí),dynamic_cast和static_cast的效果是一樣的。在進(jìn)行下行轉(zhuǎn)換時(shí),dynamic_cast具有類型檢查的功能,比static_cast更安全。當(dāng)用于多態(tài)類型時(shí),它允許任意的隱式類型轉(zhuǎn)換以及相反過程. 不過,與static_cast不同,在后一種情況里(注:即隱式轉(zhuǎn)換的相反過程),dynamic_cast 會(huì)檢查操作是否有效. 也就是說, 它會(huì)檢查轉(zhuǎn)換是否會(huì)返回一個(gè)被請求的有效的完整對象。檢測在運(yùn)行時(shí)進(jìn)行. 如果被轉(zhuǎn)換的指針不是一個(gè)被請求的有效完整的對象指針,返回值為NULL. 對于引用 類型,會(huì)拋出bad_cast異常
你說這個(gè)強(qiáng)轉(zhuǎn)有啥用,其實(shí)對于我這種菜鳥還真的沒用過,不過我知道一個(gè)問題可以使用這樣的方法解決. 給你兩個(gè)類讓你分辨那個(gè)是子類那個(gè)是父類,我們來看看是如何解決的.
#include<iostream> #include<Windows.h> #include<assert.h> using namespace std; class AA { public: virtual void fun1() { cout << "hehe" << endl; } public: int a; }; class BB :public AA { public: virtual void fun1() { cout << "heh2e" << endl; } public: int c; }; int main() { AA* q = new AA(); BB* p = new BB(); AA* a; BB* b; b = dynamic_cast<BB*>(q); if (b == NULL) { cout << "AA為基類" << endl; } else{ cout << "AA為子類" << endl; } a = dynamic_cast<AA*>(p); if (a == NULL) { cout << "BB為基類" << endl; } else { cout << "BB為子類" << endl; } system("pause"); return 0; }
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對我們的支持。
上一篇:C++中Boost.Chrono時(shí)間庫的使用方法
欄 目:C語言
下一篇:C語言中數(shù)據(jù)結(jié)構(gòu)之鏈?zhǔn)交鶖?shù)排序
本文標(biāo)題:關(guān)于C++的強(qiáng)制類型轉(zhuǎn)換淺析
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/1199.html
您可能感興趣的文章
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)式函數(shù)庫
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對數(shù)怎么表達(dá)
- 04-02c語言沒有round函數(shù) round c語言
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 01-10c語言求1+2+...+n的解決方法
- 01-10求子數(shù)組最大和的解決方法詳解
- 01-10深入理解約瑟夫環(huán)的數(shù)學(xué)優(yōu)化方法
- 01-10深入二叉樹兩個(gè)結(jié)點(diǎn)的最低共同父結(jié)點(diǎn)的詳解
- 01-10數(shù)據(jù)結(jié)構(gòu)課程設(shè)計(jì)- 解析最少換車次數(shù)的問題詳解
- 01-10c語言 跳臺(tái)階問題的解決方法


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達(dá)式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 04-02jquery與jsp,用jquery
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 08-05DEDE織夢data目錄下的sessions文件夾有什