亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

詳解C++中如何將構造函數或析構函數的訪問權限定為private

2020-01-26 14:41:30
字體:
來源:轉載
供稿:網友

今天面試被問到了這個單例模式常用到的技術手段,下面進行分析:
        很多情況下要求當前的程序中只有一個object。例如一個程序只有一個和數據庫的連接,只有一個鼠標的object。通常我們都將構造函數的聲明置于public區段,假如我們將其放入private區段中會發生什么樣的后果?這意味著什么?
        當我們在程序中聲明一個對象時,編譯器為調用構造函數(如果有的話),而這個調用將通常是外部的,也就是說它不屬于class對象本身的調用,假如構造函數是私有的,由于在class外部不允許訪問私有成員,所以這將導致編譯出錯。
        然而,對于class本身,可以利用它的static公有成員,因為它們獨立于class對象之外,不必產生對象也可以使用它們。
        此時因為構造函數被class私有化,所以我們要創建出對象,就必須能夠訪問到class的私有域;這一點只有class的成員可以做得到;但在我們建構出其對象之前,怎么能利用它的成員呢?static公有成員,它是獨立于class對象而存在的,“我們”可以訪問得到。假如在某個static函數中創建了該class的對象,并以引用或者指針的形式將其返回(這里不以對象返回,主要是構造函數是私有的,外部不能創建臨時對象),就獲得了這個對象的使用權。
下面是例子:

class OnlyHeapClass { public:   static OnlyHeapClass* GetInstance()   {     // 創建一個OnlyHeapClass對象并返回其指針     return (new OnlyHeapClass);   }   void Destroy(); private:   OnlyHeapClass() { }   ~OnlyHeapClass() {} };  int main() {   OnlyHeapClass *p = OnlyHeapClass::GetInstance();   ... // 使用*p   delete p;   return 0; } 

 
        這個例子使用了私有構造函數,GetInstance()作為OnlyHeapClass的靜態成員函數來在內存中創建對象:由于要跨函數傳遞并且不能使用值傳遞方式,所以我們選擇在堆上創建對象,這樣即使getInstance()退出,對象也不會隨之釋放,可以手動釋放。
        構造函數私有化的類的設計保證了其他類不能從這個類派生或者創建類的實例,還有這樣的用途:例如,實現這樣一個class:它在內存中至多存在一個,或者指定數量個的對象(可以在class的私有域中添加一個static類型的計數器,它的初值置為0,然后在GetInstance()中作些限制:每次調用它時先檢查計數器的值是否已經達到對象個數的上限值,如果是則產生錯誤,否則才new出新的對象,同時將計數器的值增1.最后,為了避免值復制時產生新的對象副本,除了將構造函數置為私有外,復制構造函數也要特別聲明并置為私有。
        如果將構造函數設計成Protected,也可以實現同樣的目的,但是可以被繼承。
        另外如何保證只能在堆上new一個新的類對象呢?只需把析構函數定義為私有成員。
        原因是C++是一個靜態綁定的語言。在編譯過程中,所有的非虛函數調用都必須分析完成。即使是虛函數,也需檢查可訪問性。因些,當在棧上生成對象時,對象會自動析構,也就說析構函數必須可以訪問。而堆上生成對象,由于析構時機由程序員控制,所以不一定需要析構函數。保證了不能在棧上生成對象后,需要證明能在堆上生成它。這里OnlyHeapClass與一般對象唯一的區別在于它的析構函數為私有。delete操作會調用析構函數。所以不能編譯。
        那么如何釋放它呢?答案也很簡單,提供一個成員函數,完成delete操作。在成員函數中,析構函數是可以訪問的。當然detele操作也是可以編譯通過。

void OnlyHeapClass::Destroy() {   delete this; } 

        構造函數私有化的類的設計可以保證只能用new命令在堆中來生成對象,只能動態的去創建對象,這樣可以自由的控制對象的生命周期。但是,這樣的類需要提供創建和撤銷的公共接口。
        另外重載delete,new為私有可以達到要求對象創建于棧上的目的,用placement new也可以創建在棧上。

補充:
1.為什么要自己調用呢?對象結束生存期時不就自動調用析構函數了嗎?什么情況下需要自己調用析構函數呢?  
        比如這樣一種情況,你希望在析構之前必須做一些事情,但是用你類的人并不知道, 那么你就可以重新寫一個函數,里面把要做的事情全部做完了再調用析構函數。 這樣人家只能調用你這個函數析構對象,從而保證了析構前一定會做你要求的動作。

2.什么情況下才用得著只生成堆對象呢?
        堆對象就是new出來的,相對于棧對象而言。什么情況下要new,什么情況下在棧里面 提前分配,無非就是何時該用動態,何時該用靜態生成的問題。這個要根據具體情況具體分析。比如你在一個函數里面事先知道某個對象最多只可能10個,那么你就可以 定義這個對象的一個數組。10個元素,每個元素都是一個棧對象。如果你無法確定數 字,那么你就可以定義一個這個對象的指針,需要創建的時候就new出來,并且用list 或者vector管理起來。

        類中“私有”權限的含義就是:私有成員只能在類域內被訪問,不能在類域外進行訪問。

        把析構函數定義為私有的,就阻止了用戶在類域外對析構函數的使用。這表現在如下兩個方面:

        1. 禁止用戶對此類型的變量進行定義,即禁止在棧內存空間內創建此類型的對象。要創建對象,只能用 new 在堆上進行。

        2. 禁止用戶在程序中使用 delete 刪除此類型對象。對象的刪除只能在類內實現,也就是說只有類的實現者才有可能實現對對象的 delete,用戶不能隨便刪除對象。如果用戶想刪除對象的話,只能按照類的實現者提供的方法進行。

        可見,這樣做之后大大限制了用戶對此類的使用。一般來說不要這樣做;通常這樣做是用來達到特殊的目的,比如在 singleton 的實現上。

PS:構造函數為什么不能是虛函數
另外再來說一下構造函數和虛函數的區別:
       1. 從存儲空間角度,虛函數對應一個指向vtable虛函數表的指針,這大家都知道,可是這個指向vtable的指針其實是存儲在對象的內存空間的。問題出來了,如果構造函數是虛的,就需要通過 vtable來調用,可是對象還沒有實例化,也就是內存空間還沒有,怎么找vtable呢?所以構造函數不能是虛函數。
        2. 從使用角度,虛函數主要用于在信息不全的情況下,能使重載的函數得到對應的調用。構造函數本身就是要初始化實例,那使用虛函數也沒有實際意義呀。所以構造函數沒有必要是虛函數。虛函數的作用在于通過父類的指針或者引用來調用它的時候能夠變成調用子類的那個成員函數。而構造函數是在創建對象時自動調用的,不可能通過父類的指針或者引用去調用,因此也就規定構造函數不能是虛函數。
        3. 構造函數不需要是虛函數,也不允許是虛函數,因為創建一個對象時我們總是要明確指定對象的類型,盡管我們可能通過實驗室的基類的指針或引用去訪問它但析構卻不一定,我們往往通過基類的指針來銷毀對象。這時候如果析構函數不是虛函數,就不能正確識別對象類型從而不能正確調用析構函數。
        4. 從實現上看,vbtl在構造函數調用后才建立,因而構造函數不可能成為虛函數從實際含義上看,在調用構造函數時還不能確定對象的真實類型(因為子類會調父類的構造函數);而且構造函數的作用是提供初始化,在對象生命期只執行一次,不是對象的動態行為,也沒有必要成為虛函數。
        5. 當一個構造函數被調用時,它做的首要的事情之一是初始化它的VPTR。因此,它只能知道它是“當前”類的,而完全忽視這個對象后面是否還有繼承者。當編譯器為這個構造函數產生代碼時,它是為這個類的構造函數產生代碼――既不是為基類,也不是為它的派生類(因為類不知道誰繼承它)。所以它使用的VPTR必須是對于這個類的VTABLE。而且,只要它是最后的構造函數調用,那么在這個對象的生命期內,VPTR將保持被初始化為指向這個VTABLE, 但如果接著還有一個更晚派生的構造函數被調用,這個構造函數又將設置VPTR指向它的 VTABLE,等.直到最后的構造函數結束。VPTR的狀態是由被最后調用的構造函數確定的。這就是為什么構造函數調用是從基類到更加派生類順序的另一個理由。但是,當這一系列構造函數調用正發生時,每個構造函數都已經設置VPTR指向它自己的VTABLE。如果函數調用使用虛機制,它將只產生通過它自己的VTABLE的調用,而不是最后的VTABLE(所有構造函數被調用后才會有最后的VTABLE)。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品入口免费视频一| 久久99精品久久久久久琪琪| 91精品视频在线看| 久久青草精品视频免费观看| 92裸体在线视频网站| 欧美伊久线香蕉线新在线| 亚洲自拍偷拍福利| 九九热这里只有在线精品视| 国产精品视频在线播放| 97超碰蝌蚪网人人做人人爽| 欧美亚洲日本黄色| 国产亚洲综合久久| 精品爽片免费看久久| 中文字幕亚洲情99在线| 欧美亚洲免费电影| 日韩精品视频在线观看免费| 久久精品亚洲一区| 九九精品视频在线| 国产福利成人在线| 97在线观看视频| 成人妇女淫片aaaa视频| 美女啪啪无遮挡免费久久网站| 欧美日韩在线视频一区二区| 日韩激情片免费| 亚洲区一区二区| 国产成人精品av在线| 日韩国产精品一区| 国产精品久久久久久久久影视| 欧美成人免费观看| 高潮白浆女日韩av免费看| 国产精品网站视频| 国产深夜精品福利| 欧美最近摘花xxxx摘花| 国外视频精品毛片| 动漫精品一区二区| 国产香蕉一区二区三区在线视频| 亚洲视频免费一区| 亚洲激情视频在线| 欧美人成在线视频| 国产97在线|亚洲| 色午夜这里只有精品| www.亚洲人.com| 国产精品日韩欧美综合| 97国产在线观看| 亚洲男人天堂九九视频| 亚洲jizzjizz日本少妇| 精品日本美女福利在线观看| 亚洲一区美女视频在线观看免费| 久久99热精品| 亚洲色图17p| 国产精品天天狠天天看| 欧美激情视频一区二区| 欧美日韩国产精品一区二区三区四区| 国产精品视频公开费视频| 亚洲精品女av网站| 欧美日本亚洲视频| 91久久精品国产91久久性色| 日韩av一区在线| 国产一区二区三区在线| 国产美女扒开尿口久久久| 91福利视频在线观看| 中文字幕成人精品久久不卡| 国产精品亚洲网站| 精品久久久国产精品999| 日韩欧美在线免费观看| 日韩小视频在线观看| 日本aⅴ大伊香蕉精品视频| 黄色精品一区二区| 隔壁老王国产在线精品| 国产欧美日韩精品在线观看| 91精品视频一区| 久久精品影视伊人网| 成人黄色av播放免费| 亚洲福利视频久久| 成人网在线免费看| 亚洲专区国产精品| 亚洲欧美日本伦理| 日本不卡视频在线播放| 国产欧洲精品视频| 久久av资源网站| 精品久久久久人成| 成人久久久久久久| 成人妇女免费播放久久久| 欧美电影在线观看完整版| 成人伊人精品色xxxx视频| 国产精品成熟老女人| 91精品国产综合久久久久久久久| 欧美大片在线免费观看| 欧美成人午夜免费视在线看片| 国产一区二区日韩精品欧美精品| 国产精品久久久久一区二区| 亚洲国产成人久久综合| 国产小视频国产精品| 一本一本久久a久久精品牛牛影视| 91在线播放国产| 久久久久这里只有精品| 在线观看国产精品91| 亚洲开心激情网| 亚洲人成电影在线播放| 日韩中文字幕在线视频播放| 国产在线一区二区三区| 韩国福利视频一区| 亚洲成人性视频| 国产欧美亚洲精品| 91久久精品国产91久久| 亚洲一区二区自拍| 日韩综合中文字幕| 亚洲成人亚洲激情| 在线精品播放av| 91精品久久久久久久久| 欧美亚洲另类在线| 欧美性理论片在线观看片免费| 欧美黄色片免费观看| 国产精品偷伦视频免费观看国产| 久久精品国产一区二区电影| 欧美视频专区一二在线观看| 亚洲va欧美va国产综合剧情| 美女国内精品自产拍在线播放| 国产精品揄拍500视频| 日韩专区在线播放| 久久国产精品久久久| 亚洲精品国产成人| 精品露脸国产偷人在视频| 欧美大全免费观看电视剧大泉洋| 亚洲va男人天堂| 日本一区二三区好的精华液| 国产精品久久久久国产a级| 国产欧美精品va在线观看| 精品中文字幕在线2019| 中文字幕亚洲在线| 国产欧美精品在线| 日韩高清电影好看的电视剧电影| 美女999久久久精品视频| 久久成人精品一区二区三区| 精品国产鲁一鲁一区二区张丽| 亚洲视频一区二区| 精品少妇一区二区30p| 亚洲精品美女网站| 欧美黄色片在线观看| 欧美日韩国产限制| 成人在线精品视频| 成人性生交大片免费看小说| 亚洲精品自拍第一页| 日韩av有码在线| 91精品国产91久久| 国产精品成人aaaaa网站| 欧美在线视频免费观看| 欧美性猛交丰臀xxxxx网站| 国产精品一区二区三区成人| 欧美精品在线免费| 欧美午夜片欧美片在线观看| 国产剧情日韩欧美| 日韩精品免费在线视频| 91免费在线视频| 欧美一区二区三区精品电影| 久久影院资源站| 日韩国产高清视频在线| 琪琪亚洲精品午夜在线| 精品久久香蕉国产线看观看gif| 欧美大片欧美激情性色a∨久久| 国产精品自拍网| 亚洲欧美在线磁力| 6080yy精品一区二区三区| 亚洲福利精品在线|