C++中類的轉(zhuǎn)換函數(shù)你了解嗎
只有接受一個(gè)參數(shù)(其他參數(shù)有默認(rèn)值的也算)的構(gòu)造函數(shù)才能作為轉(zhuǎn)換構(gòu)造函數(shù)。
在C++中,接受一個(gè)參數(shù)的構(gòu)造函數(shù)為將類型與該參數(shù)相同的值轉(zhuǎn)換為類提供了藍(lán)圖。因此,下面的構(gòu)造函數(shù)用于將double類型的值轉(zhuǎn)換為Stonewt類型:
Stonewt(double lbs) // double轉(zhuǎn)Stonewt的模板
也就是說,可以編寫這樣的代碼:
Stonewt myCat; // 創(chuàng)建一個(gè)Stonewt對(duì)象 myCat = 19.6; // 使用Stonewt(double)將19.6轉(zhuǎn)換為Stonewt對(duì)象
這一過程稱為隱式轉(zhuǎn)換,因?yàn)樗亲詣?dòng)進(jìn)行的,而不需要顯式強(qiáng)制類型轉(zhuǎn)換。
將構(gòu)造函數(shù)用作自動(dòng)類型轉(zhuǎn)換似乎是一項(xiàng)不錯(cuò)的特性。然而,當(dāng)程序員擁有更豐富的C++經(jīng)驗(yàn)時(shí),將發(fā)現(xiàn)這種自動(dòng)特性并非總是合乎需要的,因?yàn)檫@會(huì)導(dǎo)致意外的類型轉(zhuǎn)換。因此,C++新增了關(guān)鍵字explicit,用于關(guān)閉這種自動(dòng)特性。也就是說,可以這樣聲明構(gòu)造函數(shù):
explicit Stonewt(double lbs) // 不允許隱式轉(zhuǎn)換
這將關(guān)閉上述示例中介紹的隱式轉(zhuǎn)換,但仍然允許顯式轉(zhuǎn)換,即顯式強(qiáng)制類型轉(zhuǎn)換:
Stonewt myCat; myCat = 19.6; // 不允許,因?yàn)镾tonewt(double)聲明為explicit myCat = Stonewt(19.6); // OK,一個(gè)顯示地轉(zhuǎn)換 myCat = (Stonewt)19.6; // OK,原始地顯示類型轉(zhuǎn)換
編譯器在什么時(shí)候?qū)⑹褂肧tonewt(double)函數(shù)呢?
如果在聲明中使用了關(guān)鍵字explicit,則Stonewt(double)將只用于顯式強(qiáng)制類型轉(zhuǎn)換,否則還可以用于下面的隱式轉(zhuǎn)換:
- 將Stonewt對(duì)象初始化為double值時(shí)。如:Stonewt st(1.23);
- 將double值賦給Stonewt對(duì)象時(shí)。如:Stonewt st; st = 1.23;
- 將double值傳遞給接受Stonewt參數(shù)的函數(shù)時(shí)。如:display(Stonewt& st);-> display(1.23);
- 返回值被聲明為Stonewt的函數(shù)試圖返回double值時(shí)。如:Stonewt& go(){ return 1.23; }
- 在上述任意一種情況下,使用可轉(zhuǎn)換為double類型的內(nèi)置類型時(shí)。如:Stonewt& go() { return 123; }
可以將數(shù)字轉(zhuǎn)換為Stonewt對(duì)象??梢宰鱿喾吹霓D(zhuǎn)換嗎?也就是說,是否可以將Stonewt對(duì)象轉(zhuǎn)換為double值,就像如下所示的那樣?
Stonewt wolfe(285.7); double host = wolfe;// ??可以嗎??
可以這樣做,但不是使用構(gòu)造函數(shù)。構(gòu)造函數(shù)只用于從某種類型到類類型的轉(zhuǎn)換。要進(jìn)行相反的轉(zhuǎn)換,必須使用特殊的C++運(yùn)算符函數(shù)——轉(zhuǎn)換函數(shù)。
轉(zhuǎn)換函數(shù)是用戶定義的強(qiáng)制類型轉(zhuǎn)換,可以像使用強(qiáng)制類型轉(zhuǎn)換那樣使用它們。例如,如果定義了從Stonewt到double的轉(zhuǎn)換函數(shù),就可以使用下面的轉(zhuǎn)換:
Stonewt wolfe(285.7); double host = double (wolfe); // syntax #1 double thinker = (double) wolfe; // syntax #2 也可以讓編譯器來決定如何做: Stonewt wells(20, 3); double star = wells; // 隱式使用轉(zhuǎn)換函數(shù)
編譯器發(fā)現(xiàn),右側(cè)是Stonewt類型,而左側(cè)是double類型,因此它將查看程序員是否定義了與此匹配的轉(zhuǎn)換函數(shù)。(如果沒有找到這樣的定義,編譯器將生成錯(cuò)誤消息,指出無法將Stonewt賦給double。)
創(chuàng)建一個(gè)轉(zhuǎn)換函數(shù),轉(zhuǎn)換為typeName類型,需要使用這種形式的轉(zhuǎn)換函數(shù):
operator typeName();// 無返回類型,無參數(shù)
請(qǐng)注意以下幾點(diǎn):
- 轉(zhuǎn)換函數(shù)必須是類方法
- 轉(zhuǎn)換函數(shù)不能指定返回類型
- 轉(zhuǎn)換函數(shù)不能有參數(shù)
在類的頭文件定義:
operator int() const; operator double() const;
實(shí)現(xiàn)代碼:
......
// construct Stonewt object from stone, double values
Stonewt::Stonewt(int stn, double lbs) {
stone = stn;
pds_left = lbs;
pounds = stn * Lbs_per_stn + lbs;
}
......
// conversion functions
Stonewt::operator int() const {
return int(pounds + 0.5);
}
Stonewt::operator double() const {
return pounds;
}使用:
Stonewt poppins(9, 2.8);// 9 stone, 2.8 pounds double p_wt = poppins;// implicit conversion cout << "Convert to double => "; cout << "Poppins: " << p_wt << " pounds.\n"; cout << "Convert to int => "; cout << "Poppins: " << int(poppins) << " pounds.\n";
Convert to double => Poppins: 128.8 pounds. Convert to int => Poppins: 129 pounds.
原則上說,最好使用顯式轉(zhuǎn)換,而避免隱式轉(zhuǎn)換。在C++98中,關(guān)鍵字explicit不能用于轉(zhuǎn)換函數(shù),但C++11消除了這種限制。因此,在C++11中,可將轉(zhuǎn)換運(yùn)算符聲明為顯式的:
class Stonewt {
...
// 轉(zhuǎn)換函數(shù)
explicit operator int() const;
explicit operator double() const;
};有了explicit聲明后,在顯式強(qiáng)制轉(zhuǎn)換時(shí)才調(diào)用這些運(yùn)算符。
另一種方法是,用一個(gè)功能相同的非轉(zhuǎn)換函數(shù)替換該轉(zhuǎn)換函數(shù)即可,但僅在被顯式地調(diào)用時(shí),該函數(shù)才會(huì)執(zhí)行。也就是說,可以將:
Stonewt::operator int() { return int (pounds + 0.5); }
替換為:
int Stonewt::Stone_to_Int() { return int (pounds + 0.5); }這樣,下面的語句:
int plb = poppins; // 非法 int plb1 = (int) poppins;// 合法 int plb2 = int(poppins);// 合法 int plb3 = poppins.Stone_to_Int(); // 合法
應(yīng)謹(jǐn)慎地使用隱式轉(zhuǎn)換函數(shù)。通常,最好選擇僅在被顯式地調(diào)用時(shí)才會(huì)執(zhí)行的函數(shù)。
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Matlab實(shí)現(xiàn)繪制立體玫瑰花的示例代碼
這篇文章主要介紹了如何利用Matlab實(shí)現(xiàn)繪制更立體的玫瑰花,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Matlab有一定的幫助,需要的可以參考一下2023-02-02
C++中引用(&)的用法與應(yīng)用實(shí)例分析
引用是C++引入的新語言特性,是C++常用的一個(gè)重要內(nèi)容之一,正確、靈活地使用引用,可以使程序簡潔、高效。故在本篇中我將對(duì)引用進(jìn)行詳細(xì)討論,希望對(duì)大家更好地理解和使用引用起到拋磚引玉的作用2013-09-09
vscode配置遠(yuǎn)程開發(fā)環(huán)境并遠(yuǎn)程調(diào)試運(yùn)行C++代碼的教程
這篇文章主要介紹了vscode配置遠(yuǎn)程開發(fā)環(huán)境并遠(yuǎn)程調(diào)試運(yùn)行C++代碼的教程,本文通過截圖實(shí)例相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04

