c++隱式類型轉(zhuǎn)換示例分享
/*=============================================================================
# FileName: explicit_try.cc
# Desc: 驗證含有一個參數(shù)的非explicit構(gòu)造函數(shù)是否可以拷貝初始化
=============================================================================*/
#include <iostream>
#include <string>
#include <vector>
using std::cout;
using std::cin;
using std::endl;
using std::string;
using std::vector;
class People {
public:
People() = default;
People(string s):name(s) { }
string getName() const { return name; }
static vector<string> &getVector() { return name_arr; }
//隱式類型轉(zhuǎn)換,用string生成一個臨時量,因此可以綁定到const形參上
static void addToVector(const People &p) {
name_arr.push_back(p.getName());
}
private:
string name = "";
static vector<string> name_arr;
};
vector<string> People::name_arr = {};
int main(int argc, const char *argv[])
{
People p;
cout << "p :" << endl;
cout << p.getName() << endl;
People tom("tom");
People::addToVector(tom);
string Bob = "Bob";
People::addToVector(Bob);//隱式類型轉(zhuǎn)換
//People::addToVector("Bob");//只允許一步的隱式類型轉(zhuǎn)換
vector<string> v = People::getVector();
cout << "name_arr:" << endl;
for (const auto &p : v) {
cout << p << " ";
}
cout << endl;
string myName = "guo";
People guo = myName; //隱式類型轉(zhuǎn)換允許拷貝初始化形式的轉(zhuǎn)換
cout << guo.getName() << endl;
return 0;
}
下面再來一個例子
#include <string>
#include <iostream>
using namespace std;
class Fruit //定義一個類,名字叫Fruit
{
string name; //定義一個name成員
string colour; //定義一個colour成員
public:
bool isSame(const Fruit &otherFruit) //期待的形參是另一個Fruit類對象,測試是否同名
{
return name == otherFruit.name;
}
void print() //定義一個輸出名字的成員print()
{
cout<<colour<<" "<<name<<endl;
}
Fruit(const string &nst,const string &cst = "green"):name(nst),colour(cst){} //構(gòu)造函數(shù)
Fruit(){}
};
int main()
{
Fruit apple("apple");
Fruit orange("orange");
cout<<"apple = orange ?: "<<apple.isSame(orange)<<endl; //沒有問題,肯定不同
cout<<"apple = /"apple/" ?:"<<apple.isSame(string("apple")); //用一個string做形參?
return 0;
}
你會發(fā)現(xiàn)最后的使用上,我們用一個string類型作一個期待Fruit類形參的函數(shù)的參數(shù),結(jié)果竟然得出了是true(1),不要感到奇怪,這就是我現(xiàn)在要講的東西,隱式類類型轉(zhuǎn)換:“可以用單個實參來調(diào)用的構(gòu)造函數(shù)定義了從形參類型到該類型的一個隱式轉(zhuǎn)換。”(C++ Primer)首先要單個實參,你可以把構(gòu)造函數(shù)colour的默認(rèn)實參去掉,也就是定義一個對象必須要兩個參數(shù)的時候,文件編譯不能通過。然后滿足這個條件后,系統(tǒng)就知道怎么轉(zhuǎn)換了,不過這里比較嚴(yán)格:)以前我們構(gòu)造對象的時候Fruit apple("apple")其實也已經(jīng)有了一個轉(zhuǎn)換,從const char *的C字符串格式,轉(zhuǎn)為string,在這里,你再apple.isSame("apple")的話,蠢系統(tǒng)不懂得幫你轉(zhuǎn)換兩次,所以你必須要用string()來先強制轉(zhuǎn)換,然后系統(tǒng)才知道幫你從string隱式轉(zhuǎn)換為Fruit,當(dāng)然其實你自己也可以幫他完成。cout<<"apple = /"apple/" ?:"<<apple.isSame(Fruit("apple"));這樣。參考例子1.2 :Fruit apple = Fruit("apple"); //定義一個Fruit類對象apple。也就是這樣轉(zhuǎn)換的。不過這就叫顯式轉(zhuǎn)換了,我們不標(biāo)出來,系統(tǒng)幫我們完成的,叫隱式的貝。這里要說的是,假如你顯示轉(zhuǎn)換就可以不管有多少參數(shù)了,比如在前面提到的必須需要兩個參數(shù)的構(gòu)造函數(shù)時的例子。
例:
#include <string>
#include <iostream>
using namespace std;
class Fruit //定義一個類,名字叫Fruit
{
string name; //定義一個name成員
string colour; //定義一個colour成員
public:
bool isSame(const Fruit &otherFruit) //期待的形參是另一個Fruit類對象,測試是否同名
{
return name == otherFruit.name;
}
void print() //定義一個輸出名字的成員print()
{
cout<<colour<<" "<<name<<endl;
}
Fruit(const string &nst,const string &cst):name(nst),colour(cst){} //構(gòu)造函數(shù)
Fruit(){}
};
int main()
{
Fruit apple("apple","green");
Fruit orange("orange","yellow");
cout<<"apple = orange ?: "<<apple.isSame(orange)<<endl; //沒有問題,肯定不同
cout<<"apple = /"apple/" ?:"<<apple.isSame(Fruit("apple","green")); //顯式轉(zhuǎn)換
return 0;
}
在你不想隱式轉(zhuǎn)換,以防用戶誤操作怎么辦?C++提供了一種抑制構(gòu)造函數(shù)隱式轉(zhuǎn)換的辦法,就是在構(gòu)造函數(shù)前面加explicit關(guān)鍵字,你試試就知道,那時你再希望隱式轉(zhuǎn)換就會導(dǎo)致編譯失敗,但是,要說明的是,顯式轉(zhuǎn)換還是可以進(jìn)行。
相關(guān)文章
如何通過wrap malloc定位C/C++的內(nèi)存泄漏問題
用C/C++開發(fā)的程序執(zhí)行效率很高,但卻經(jīng)常受到內(nèi)存泄漏的困擾。本文提供一種通過wrap malloc查找memory leak的思路。2021-05-05