有關(guān)C++中類類型轉(zhuǎn)換操作符總結(jié)(必看篇)
實(shí)例如下:
class SmallInt { public: SmallInt(int i = 0): val(i) { if (i < 0 || i > 255) throw std::out_of_range("Bad SmallInt initializer"); } operator int() const { return val; } private: std::size_t val; };
轉(zhuǎn)換函數(shù)采用如下通用形式:
operator type();
type表示內(nèi)置類型名、類類型名或由類型別名定義的名字。對(duì)任何可作為函數(shù)返回類型的類型(除了 void 之外)都可以定義轉(zhuǎn)換函數(shù)。一般而言,不允許轉(zhuǎn)換為數(shù)組或函數(shù)類型,轉(zhuǎn)換為指針類型(數(shù)據(jù)和函數(shù)指針)以及引用類型是可以的。轉(zhuǎn)換函數(shù)必須是成員函數(shù),不能指定返回類型,并且形參表必須為空。operator int 返回一個(gè) int 值;如果定義 operator Sales_item,它將返回一個(gè) Sales_item 對(duì)象,諸如此類。轉(zhuǎn)換函數(shù)一般不應(yīng)該改變被轉(zhuǎn)換的對(duì)象。因此,轉(zhuǎn)換操作符通常應(yīng)定義為 const 成員。
SmallInt si;
double dval;
si >= dval // si converted to int and then convert todouble
優(yōu)點(diǎn):類類型轉(zhuǎn)換可能是實(shí)現(xiàn)和使用類的一個(gè)好處。通過(guò)為 SmallInt 定義到int 的轉(zhuǎn)換,能夠更容易實(shí)現(xiàn)和使用 SmallInt 類。int 轉(zhuǎn)換使 SmallInt 的用戶能夠?qū)?SmallInt 對(duì)象使用所有算術(shù)和關(guān)系操作符,而且,用戶可以安全編寫(xiě)將 SmallInt 和其他算術(shù)類型混合使用的表達(dá)式。定義一個(gè)轉(zhuǎn)換操作符就能代替定義 48個(gè)(或更多)重載操作符,類實(shí)現(xiàn)者的工作就簡(jiǎn)單多了。
缺點(diǎn):二義性
class SmallInt { public: SmallInt(int= 0); SmallInt(double); //Usually it is unwise to define conversions to multiple arithmetic types operatorint() const { return val; } operatordouble() const { return val; } private: std::size_tval; }; void compute(int); void fp_compute(double); void extended_compute(long double); SmallInt si; compute(si); // SmallInt::operator int() const fp_compute(si); // SmallInt::operator double() const extended_compute(si); // error: ambiguous
對(duì) extended_compute 的調(diào)用有二義性??梢允褂萌我晦D(zhuǎn)換函數(shù),但每個(gè)都必須跟上一個(gè)標(biāo)準(zhǔn)轉(zhuǎn)換來(lái)獲得 long double,因此,沒(méi)有一個(gè)轉(zhuǎn)換比其他的更好,調(diào)用具有二義性。
如果兩個(gè)轉(zhuǎn)換操作符都可用在一個(gè)調(diào)用中,而且在轉(zhuǎn)換函數(shù)之后存在標(biāo)準(zhǔn)轉(zhuǎn)換,則根據(jù)該標(biāo)準(zhǔn)轉(zhuǎn)換的類別選擇最佳匹配。若無(wú)最佳匹配,就會(huì)出現(xiàn)二義性。
再比如:
可能存在兩個(gè)轉(zhuǎn)換操作符,也可能存在兩個(gè)構(gòu)造函數(shù)可以用來(lái)將一個(gè)值轉(zhuǎn)換為目標(biāo)類型。
考慮 manip 函數(shù),它接受一個(gè) SmallInt 類型的實(shí)參:
void manip(const SmallInt &); double d; int i; long l; manip(d); // ok: use SmallInt(double) to convert theargument manip(i); // ok: use SmallInt(int) to convert theargument manip(l); // error: ambiguous
第三個(gè)調(diào)用具有二義性。沒(méi)有構(gòu)造函數(shù)完全匹配于 long。使用每一個(gè)構(gòu)造函
數(shù)之前都需要對(duì)實(shí)參進(jìn)行轉(zhuǎn)換:
1. 標(biāo)準(zhǔn)轉(zhuǎn)換(從 long 到double)后跟 SmallInt(double)。
2. 標(biāo)準(zhǔn)轉(zhuǎn)換(從 long 到int)后跟 SmallInt(int)。
這些轉(zhuǎn)換序列是不能區(qū)別的,所以該調(diào)用具有二義性。
當(dāng)兩個(gè)類定義了相互轉(zhuǎn)換時(shí),很可能存在二義性:
class Integral; class SmallInt { public: SmallInt(Integral);// convert from Integral to SmallInt }; class Integral { public: operatorSmallInt() const; // convert from Integral to SmallInt }; void compute(SmallInt); Integral int_val; compute(int_val); // error: ambiguous
實(shí)參 int_val 可以用兩種不同方式轉(zhuǎn)換為 SmallInt 對(duì)象,編譯器可以使
用接受 Integral 對(duì)象的構(gòu)造函數(shù),也可以使用將 Integral 對(duì)象轉(zhuǎn)換為
SmallInt 對(duì)象的 Integral 轉(zhuǎn)換操作。因?yàn)檫@兩個(gè)函數(shù)沒(méi)有高下之分,所以這
個(gè)調(diào)用會(huì)出錯(cuò)。
在這種情況下,不能用顯式類型轉(zhuǎn)換來(lái)解決二義性——顯式類型轉(zhuǎn)換本身既可以使用轉(zhuǎn)換操作又可以使用構(gòu)造函數(shù),相反,需要顯式調(diào)用轉(zhuǎn)換操作符或構(gòu)造函數(shù):
compute(int_val.operator SmallInt()); // ok: useconversion operator compute(SmallInt(int_val)); // ok: use SmallInt constructor
改變構(gòu)造函數(shù)以接受 const Integral 引用:
class SmallInt { public: SmallInt(constIntegral&); };
則對(duì)compute(int_val) 的調(diào)用不再有二義性!原因在于使用 SmallInt構(gòu)造函數(shù)需要將一個(gè)引用綁定到 int_val,而使用 Integral 類的轉(zhuǎn)換操作符可以避免這個(gè)額外的步驟。這一小小區(qū)別足以使我們傾向于使用轉(zhuǎn)換操作符。
顯式強(qiáng)制轉(zhuǎn)換消除二義性
class SmallInt { public: // Usually it is unwise to define conversions tomultiple arithmetic types operatorint() const { return val; } operatordouble() const { return val; } // ... private: std::size_tval; }; void compute(int); void compute(double); void compute(long double); SmallInt si; compute(si); // error: ambiguous
可以利用顯式強(qiáng)制轉(zhuǎn)換來(lái)消除二義性:
compute(static_cast<int>(si)); // ok: convertand call compute(int)
顯式構(gòu)造函數(shù)調(diào)用消除二義性
class SmallInt { public: SmallInt(int= 0); }; class Integral { public: Integral(int= 0); }; void manip(const Integral&); void manip(const SmallInt&); manip(10); // error: ambiguous
可以用顯示構(gòu)造函數(shù)消除二義性:
manip(SmallInt(10)); // ok: call manip(SmallInt) manip(Integral(10)); // ok: call manip(Integral)
標(biāo)準(zhǔn)轉(zhuǎn)換優(yōu)于類類型轉(zhuǎn)換
class LongDouble { public: LongDouble(double ); //… }; void calc( int ); void calc( LongDouble ); double dval; calc( dval ); // which function
最佳可行函數(shù)是voidcalc(int), 調(diào)用此函數(shù)的轉(zhuǎn)換為:將實(shí)參double類型轉(zhuǎn)換為int類型的,為標(biāo)準(zhǔn)轉(zhuǎn)換;調(diào)用voidcalc( LongDouble)函數(shù)時(shí),將實(shí)參從double轉(zhuǎn)換為L(zhǎng)ongDouble類型,為類類型轉(zhuǎn)換,因?yàn)闃?biāo)準(zhǔn)轉(zhuǎn)換優(yōu)于類類型轉(zhuǎn)換,所以第一個(gè)函數(shù)為最佳可行函數(shù)。
以上這篇有關(guān)C++中類類型轉(zhuǎn)換操作符總結(jié)(必看篇)就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持我們。
上一篇:c++實(shí)現(xiàn)的常見(jiàn)緩存算法和LRU
欄 目:C語(yǔ)言
下一篇:簡(jiǎn)單談?wù)勱P(guān)于C++中大隨機(jī)數(shù)的問(wèn)題
本文標(biāo)題:有關(guān)C++中類類型轉(zhuǎn)換操作符總結(jié)(必看篇)
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/1821.html
您可能感興趣的文章
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 01-10深入理解C++中常見(jiàn)的關(guān)鍵字含義
- 01-10使用C++實(shí)現(xiàn)全排列算法的方法詳解
- 01-10c++中inline的用法分析
- 01-10用C++實(shí)現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實(shí)現(xiàn)與遞歸實(shí)現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解
- 01-10深入C/C++浮點(diǎn)數(shù)在內(nèi)存中的存儲(chǔ)方式詳解
- 01-10深入理解C/C++混合編程


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-10delphi制作wav文件的方法
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 04-02jquery與jsp,用jquery
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)