構(gòu)造函數(shù)定義為private或者protected的好處
將構(gòu)造函數(shù),析構(gòu)函數(shù)聲明為私有和保護(hù)的,那么對(duì)象如何創(chuàng)建?已經(jīng)不能從外部調(diào)用構(gòu)造函數(shù)了,但是對(duì)象必須被構(gòu)造,應(yīng)該如何解決,麻煩大家?guī)兔φf(shuō)明,關(guān)于構(gòu)造,析構(gòu)函數(shù)聲明為私有和保護(hù)時(shí)的用法??? 提出這個(gè)問(wèn)題,說(shuō)明你已經(jīng)對(duì)c++有所思考了。
從語(yǔ)法上來(lái)講,一個(gè)函數(shù)被聲明為protected或者private,那么這個(gè)函數(shù)就不能從“外部”直接被調(diào)用了。
對(duì)于protected的函數(shù),子類(lèi)的“內(nèi)部”的其他函數(shù)可以調(diào)用之。
而對(duì)于private的函數(shù),只能被本類(lèi)“內(nèi)部”的其他函數(shù)說(shuō)調(diào)用。
語(yǔ)法上就是這么規(guī)定的,你肯定也知道的咯。
那么為什么有時(shí)候?qū)?gòu)造函數(shù)或者析構(gòu)函數(shù)聲明為protected的或者private的?
通常使用的場(chǎng)景如下:
1。如果你不想讓外面的用戶直接構(gòu)造一個(gè)類(lèi)(假設(shè)這個(gè)類(lèi)的名字為A)的對(duì)象,而希望用戶只能構(gòu)造這個(gè)類(lèi)A的子類(lèi),那你就可以將類(lèi)A的構(gòu)造函數(shù)/析構(gòu)函數(shù)聲明為protected,而將類(lèi)A的子類(lèi)的構(gòu)造函數(shù)/析構(gòu)函數(shù)聲明為public。例如:
class A
{ protected: A(){}
public: ....
};
calss B : public A
{ public: B(){}
....
};
A a; // error
B b; // ok
2. 如果將構(gòu)造函數(shù)/析構(gòu)函數(shù)聲明為private,那只能這個(gè)類(lèi)的“內(nèi)部”的函數(shù)才能構(gòu)造這個(gè)類(lèi)的對(duì)象了。這里所說(shuō)的“內(nèi)部”不知道你是否能明白,下面舉個(gè)例子吧。
class A
{
private:
A(){ }
~A(){ }
public:
void Instance()//類(lèi)A的內(nèi)部的一個(gè)函數(shù)
{
A a;
}
};
上面的代碼是能通過(guò)編譯的。上面代碼里的Instance函數(shù)就是類(lèi)A的內(nèi)部的一個(gè)函數(shù)。Instance函數(shù)體里就構(gòu)建了一個(gè)A的對(duì)象。
但是,這個(gè)Instance函數(shù)還是不能夠被外面調(diào)用的。為什么呢?
如果要調(diào)用Instance函數(shù),必須有一個(gè)對(duì)象被構(gòu)造出來(lái)。但是構(gòu)造函數(shù)被聲明為private的了。外部不能直接構(gòu)造一個(gè)對(duì)象出來(lái)。
A aObj; // 編譯通不過(guò)
aObj.Instance();
但是,如果Instance是一個(gè)static靜態(tài)函數(shù)的話,就可以不需要通過(guò)一個(gè)對(duì)象,而可以直接被調(diào)用。如下:
class A
{
private:
A():data(10){ cout << "A" << endl; }
~A(){ cout << "~A" << endl; }
public:
static A& Instance()
{
static A a;
return a;
}
void Print()
{
cout << data << endl;
}
private:
int data;
};
A& ra = A::Instance();
ra.Print();
上面的代碼其實(shí)是設(shè)計(jì)模式singleton模式的一個(gè)簡(jiǎn)單的C++代碼實(shí)現(xiàn)。
還有一個(gè)情況是:通常將拷貝構(gòu)造函數(shù)和operator=(賦值操作符重載)聲明成private,但是沒(méi)有實(shí)現(xiàn)體。
這個(gè)的目的是禁止一個(gè)類(lèi)的外部用戶對(duì)這個(gè)類(lèi)的對(duì)象進(jìn)行復(fù)制動(dòng)作。
細(xì)節(jié)請(qǐng)看《effective C++》里面的條款27。
- 成員初始化列表與構(gòu)造函數(shù)體中的區(qū)別詳細(xì)解析
- c++基礎(chǔ)語(yǔ)法:構(gòu)造函數(shù)與析構(gòu)函數(shù)
- c++基礎(chǔ)語(yǔ)法:構(gòu)造函數(shù)初始化列表
- C++中拷貝構(gòu)造函數(shù)的總結(jié)詳解
- C++拷貝構(gòu)造函數(shù)(深拷貝與淺拷貝)詳解
- 深入解析C++中的構(gòu)造函數(shù)和析構(gòu)函數(shù)
- 深入解析C++中的虛函數(shù)與多態(tài)
- 淺析C++中的虛函數(shù)
- c++ 虛函數(shù)與純虛函數(shù)的區(qū)別(深入分析)
- 構(gòu)造函數(shù)不能聲明為虛函數(shù)的原因及分析
相關(guān)文章
C語(yǔ)言用遞歸函數(shù)對(duì)素?cái)?shù)進(jìn)行判斷流程
素?cái)?shù)判斷是編程語(yǔ)言學(xué)習(xí)過(guò)程中一個(gè)老生常談的話題,而它的實(shí)現(xiàn)也有多種算法,包括經(jīng)典的試除法(以及試除法的幾種優(yōu)化),進(jìn)階的素?cái)?shù)表篩選法,埃拉托斯特尼篩法和歐拉篩法(以及它們的優(yōu)化)等。對(duì)以上算法感興趣的朋友們,不妨搜索“素?cái)?shù)判斷的N種境界”來(lái)學(xué)習(xí)了解2022-09-09C++的std::transform()的實(shí)現(xiàn)
在 C++ 標(biāo)準(zhǔn)庫(kù)中,std::transform() 是一個(gè)非常有用的算法函數(shù),它能夠?qū)⒔o定范圍中的每個(gè)元素進(jìn)行變換,并將變換后的結(jié)果存儲(chǔ)到另一個(gè)范圍中,本文就詳細(xì)的介紹一下具體用法,感興趣的可以了解一下2023-08-08Linux系統(tǒng)中C語(yǔ)言編程創(chuàng)建函數(shù)fork()執(zhí)行解析
最近在看進(jìn)程間的通信,看到了fork()函數(shù),雖然以前用過(guò),這次經(jīng)過(guò)思考加深了理解。現(xiàn)總結(jié)如下2013-04-04C++中的四個(gè)默認(rèn)成員函數(shù)與運(yùn)算符重載詳解
這篇文章主要給大家介紹了關(guān)于C++中四個(gè)默認(rèn)成員函數(shù)與運(yùn)算符重載的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)跟著小編一起學(xué)習(xí)學(xué)習(xí)吧。2017-08-08C語(yǔ)言數(shù)據(jù)結(jié)構(gòu) link 鏈表反轉(zhuǎn)的實(shí)現(xiàn)
這篇文章主要介紹了C語(yǔ)言數(shù)據(jù)結(jié)構(gòu) link 鏈表反轉(zhuǎn)的實(shí)現(xiàn)的相關(guān)資料,希望通過(guò)本文能幫助到大家,需要的朋友可以參考下2017-09-09