前言
公司主營業(yè)務:成都做網(wǎng)站、成都網(wǎng)站建設(shè)、成都外貿(mào)網(wǎng)站建設(shè)、移動網(wǎng)站開發(fā)等業(yè)務。幫助企業(yè)客戶真正實現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競爭能力。創(chuàng)新互聯(lián)是一支青春激揚、勤奮敬業(yè)、活力青春激揚、勤奮敬業(yè)、活力澎湃、和諧高效的團隊。公司秉承以“開放、自由、嚴謹、自律”為核心的企業(yè)文化,感謝他們對我們的高要求,感謝他們從不同領(lǐng)域給我們帶來的挑戰(zhàn),讓我們激情的團隊有機會用頭腦與智慧不斷的給客戶帶來驚喜。創(chuàng)新互聯(lián)推出高州免費做網(wǎng)站回饋大家。
大家都知道類的繼承規(guī)則:
1、派生類自動包含基類的所有成員。但對于基類的私有成員,派生類雖然繼承了,但是不能在派生類中訪問。
2、所有的類都是按照繼承鏈從頂層基類開始向下順序構(gòu)造。最頂層的基類是System.Object
類,所有的類都隱式派生于它。只要記住這條規(guī)則,就能理解派生類在實例化時對構(gòu)造函數(shù)的調(diào)用過程。
不知道大家在使用繼承的過程中有木有遇到過調(diào)用構(gòu)造函數(shù)時沒有按照我們預期的那樣執(zhí)行呢?一般情況下,出現(xiàn)這樣的問題往往是因為類繼承結(jié)構(gòu)中的某個基類沒有被正確實例化,或者沒有正確給基類構(gòu)造函數(shù)提供信息,如果理解在對象生命周期的這個階段發(fā)生的事情,將更利于解決此類問題。
為了實例化派生的類,必須先實例化它的基類。而要實例化這個基類。又必須要實例化這個基類的基類,這樣一直到實例化System.Object
(所有類的跟)為止,結(jié)果無論使用什么構(gòu)造函數(shù)實例化一個類,總是首先調(diào)用System.Object.Object()
.
下面一個示例演示執(zhí)行順序:
基類:
public class MyBaseClass { public MyBaseClass() { Console.WriteLine("I am MyBaseClass()"); } public MyBaseClass(int i) { Console.WriteLine("I am MyBaseClass(int i)"); } }
派生類:
public MyDerivedClass() { Console.WriteLine("I am MyDerivedCalss()"); } public MyDerivedClass(int i) { Console.WriteLine("I am MyDerivedClass(int i)"); } public MyDerivedClass(int i,int j) { Console.WriteLine("I am MyDerivedClass(int i,int j)"); }
接下來我們在Main函數(shù)中以不帶參數(shù)的構(gòu)造函數(shù)實例化MyDerivedClass:
MyDerivedClass myObj = new MyDerivedClass();
運行程序,控制臺輸出如下:
從結(jié)果可以看出,執(zhí)行順序先是基類構(gòu)造的函數(shù),接下來才是派生類的構(gòu)造函數(shù),即
1.執(zhí)行System.Object.Object()
構(gòu)造函數(shù)(Object比較特殊,所有類的基類,一般可以不考慮,但是得知道它也是被執(zhí)行了的)
2.執(zhí)行MyBaseClass.MyBaseClass()
構(gòu)造函數(shù)
3.執(zhí)行MyDerivedClass.MyDerivedClass()
構(gòu)造函數(shù)
如果我們以帶一個參數(shù)的構(gòu)造函數(shù)實例化MyDerivedClass:
MyDerivedClass myObj = new MyDerivedClass(4);
運行程序,控制臺輸出如下:
可以看出執(zhí)行順序如下:
1.執(zhí)行System.Object.Object()
構(gòu)造函數(shù)
2.執(zhí)行MyBaseClass.MyBaseClass()
構(gòu)造函數(shù)
3.執(zhí)行MyDerivedClass.MyDerivedClass(int i)
構(gòu)造函數(shù)
同理如果我們以帶兩個參數(shù)的構(gòu)造函數(shù)實例化MyDerivedClass
MyDerivedClass myObj = new MyDerivedClass(4,8);
運行程序,控制臺輸出如下:
可以看出執(zhí)行順序如下:
1.執(zhí)行System.Object.Object()
構(gòu)造函數(shù)
2.執(zhí)行MyBaseClass.MyBaseClass()
構(gòu)造函數(shù)
3.執(zhí)行MyDerivedClass.MyDerivedClass(int i,int j)
構(gòu)造函數(shù)
大多數(shù)情況下這個都能正常工作,但是有時我們需要對發(fā)生的事件進行更多的控制。比如我們想得到如下所示的執(zhí)行順序:
1.執(zhí)行System.Object.Object()構(gòu)造函數(shù)
2.執(zhí)行MyBaseClass.MyBaseClass(int i)
構(gòu)造函數(shù)
3.執(zhí)行MyDerivedClass.MyDerivedClass(int i,int j)
構(gòu)造函數(shù)
使用這個順序,可以把使用int i參數(shù)的代碼放到MyBaseClass(int i)
中,MyDerivedClass(int i,int j)
只需要處理int j(假設(shè)int i參數(shù)在MyBaseClass和 MyDerivedClass里含義是一樣的)
為此,只需要使用構(gòu)造函數(shù)初始化器,把代碼放到方法定義的冒號后面,如在派生類的構(gòu)造函數(shù)中指定所使用的基類的構(gòu)造函數(shù),如下所示:
public MyDerivedClass(int i,int j) : base(i) { Console.WriteLine("I am MyDerivedClass(int i,int j)"); }
其中,base關(guān)鍵字指定在實例化過程中使用具有指定參數(shù)的構(gòu)造函數(shù)。這里使用了int參數(shù),其值通過i傳遞給MyDerivedClass構(gòu)造函數(shù),所以將使用MyBaseClass(int i)
,這樣就不會調(diào)用MyBaseClass()
了,我們重新執(zhí)行下前面兩個參數(shù)的實例化代碼,就可以看出執(zhí)行結(jié)果確實如此:
除了base關(guān)鍵字,還可以使用this關(guān)鍵字用作構(gòu)造函數(shù)初始化器,這個關(guān)鍵字指定在調(diào)用指定的構(gòu)造函數(shù)前,實例化過程對當前類使用非默認的構(gòu)造函數(shù)。例如:
public MyDerivedClass():this(5,6) { Console.WriteLine("I am MyDerivedCalss()"); }
使用MyDerivedCalss()
構(gòu)造函數(shù)實例化,執(zhí)行順序是:
1.執(zhí)行System.Object.Object()
構(gòu)造函數(shù)
2.執(zhí)行MyBaseClass.MyBaseClass(int i)
構(gòu)造函數(shù)
3.執(zhí)行MyDerivedClass.MyDerivedClass(int i,int j)
構(gòu)造函數(shù)
4.執(zhí)行MyDerivedClass.MyDerivedClass()
構(gòu)造函數(shù)
總之呢,無論派生類上使用什么樣的構(gòu)造函數(shù)(默認的or不是默認的),除非明確指定(如使用base關(guān)鍵字),否則就先調(diào)用用基類的默認構(gòu)造函數(shù)。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對創(chuàng)新互聯(lián)的支持。