c++隱式類型轉(zhuǎn)換存在的問題解析
什么是隱式轉(zhuǎn)換:
c++中的基本類型并非完全對立,部分類型之間是可以進(jìn)行隱式轉(zhuǎn)換的,所謂隱式轉(zhuǎn)換,是指不需要用戶干預(yù),編譯器私下進(jìn)行的類型轉(zhuǎn)換行為,很多時候用戶都不知道具體進(jìn)行了哪些轉(zhuǎn)換
為什么要進(jìn)行隱式轉(zhuǎn)換:
隱式轉(zhuǎn)換可以讓程序員在兩個不同類型的數(shù)據(jù)直接進(jìn)行操作,而不用自行轉(zhuǎn)換類型,隱式轉(zhuǎn)換給程序開發(fā)者帶來了不小的便捷
隱式轉(zhuǎn)換的原則:
- 基本數(shù)據(jù)類型的轉(zhuǎn)換以取值范圍作為轉(zhuǎn)換基礎(chǔ)(保證精度不丟失)
- 隱式轉(zhuǎn)換發(fā)生在從小 ——> 大的轉(zhuǎn)換中。如char轉(zhuǎn)換為int,int轉(zhuǎn)換成double
目標(biāo)代碼
旨在弄懂下面的代碼,明確變量a1,a2,a3在創(chuàng)建時編譯器究竟干了那些事:
#include<iostream> using namespace std; class A{ public: int x; A() {cout<<"A()"<<endl;} A(int i) : x(i){cout<<"A(int i)"<<endl;} A(const A &ra) : x(ra.x) {cout<<"A(const A&)"<<endl;} void operator=(const A&){cout<<"operator="<<endl;} }; int main(){ cout<<"the assignment of a1 : "<<endl; A a1; a1 = 2; cout<<"the copy initialization of a2 : "<<endl; A a2 = 2; cout<<"the direct initialization of a3 :"<<endl; A a3(a1); return 0; }
運(yùn)行結(jié)果:
構(gòu)造函數(shù)定義的隱式類型轉(zhuǎn)換
任何只接受一個參數(shù)的構(gòu)造函數(shù),都隱式地定義了由該參數(shù)向該類型的隱式類型轉(zhuǎn)換
如A(int i)
定義了一個由int向A的隱式類型轉(zhuǎn)換
所以,在任何使用A對象的地方,可以用一個int代替,此時,int會轉(zhuǎn)換為一個A類型臨時變量
如對a1變量的賦值操作:
A a1; //聲明a1,a1被默認(rèn)初始化 a1 = 2; //2轉(zhuǎn)換為A類型的臨時變量,對a1進(jìn)行賦值操作
對于隱式類型轉(zhuǎn)換,需要注意兩點(diǎn):
隱式類型轉(zhuǎn)換只允許一步轉(zhuǎn)換
class B{ public: string B_s; B() = default; B(string s) : B_s(s){}; }; int main(){ B b1,b2; //錯誤:char*->string->B,進(jìn)行了兩步轉(zhuǎn)換 b1 = "hello"; b2 = string("hello"); return 0; }
接受隱式類型轉(zhuǎn)換得到的對象的函數(shù),參數(shù)傳遞方式必須是const引用傳遞
因?yàn)閏++中,一般不修改臨時對象,所以臨時對象只能傳遞給const引用。
分析a1
A a1
:
a1進(jìn)行默認(rèn)初始化,調(diào)用默認(rèn)構(gòu)造函數(shù)A()
a1 = 2
- 字面量2隱式轉(zhuǎn)換為A類型的臨時對象
- 該臨時對象通過拷貝運(yùn)算符
operator=
拷貝給a1 - 因?yàn)槭桥R時對象,所以
operator=
必須接受const引用,否則造成編譯錯誤
分析a2
A a2 = 2
- 字面量2隱式轉(zhuǎn)換為A類型的臨時對象
- 用臨時對象來拷貝初始化a2,調(diào)用拷貝構(gòu)造函數(shù)
A(const A&)
,相當(dāng)于A a2(A(2))
- 因?yàn)槭桥R時對象,所以拷貝構(gòu)造函數(shù)
A(const A&)
必須接受const引用,否則造成編譯錯誤
特別注意
編譯器會將A a2(A(2))
優(yōu)化為A a2(2)
所以程序輸出“A(int i)”,而不是“A(const A&)”
但是底層仍然調(diào)用了A(const A&),所以如果把A(const A&)改為A(A&),會造成編譯錯誤 error: cannot bind non-const lvalue reference of type 'A&' to an rvalue of type 'A'
這提示我們,在編寫c++程序時,如果不改變對象的值,那么習(xí)慣性地采用const引用會避免許多難解的編譯錯誤
分析a3
用a1直接初始化a3,調(diào)用A(const A&)
到此這篇關(guān)于c++隱式類型轉(zhuǎn)換存在的陷阱的文章就介紹到這了,更多相關(guān)c++隱式類型轉(zhuǎn)換內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++中小數(shù)點(diǎn)輸出格式(實(shí)例代碼)
下面小編就為大家?guī)硪黄狢++中小數(shù)點(diǎn)輸出格式(實(shí)例代碼)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-06-06C語言中#pragma?pack(1)的用法與注意點(diǎn)
#pragma用于指示編譯器完成一些特定的動作,下面這篇文章主要給大家介紹了關(guān)于C語言中#pragma?pack(1)的用法與注意點(diǎn)的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-02-02C++ 多線程編程建議之 C++ 對多線程/并發(fā)的支持(下)
這篇文章主要介紹的是 C++ 多線程編程建議之 C++ 對多線程/并發(fā)的支持的相關(guān)資料,承接前文 現(xiàn)代 C++ 對多線程/并發(fā)的支持,接下來我們看看回發(fā)生什么吧2021-10-10c++ builder TreeView控件節(jié)點(diǎn)遍歷代碼
這篇文章介紹了c++ builder TreeView控件節(jié)點(diǎn)遍歷代碼,有需要的朋友可以參考一下2013-09-09C語言實(shí)現(xiàn)餐廳點(diǎn)餐系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)餐廳點(diǎn)餐系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-06-06