今天在CSDN上看到有人問某個類及其衍生類別的執行過程,並問輸出結果: 1using System;
2using System.Collections.Generic;
3using System.Text;
4
5namespace ConsoleApplication1 {
6 class Program {
7 static void Main(string[] args) {
8 A a = new B();
9 Console.ReadLine();
10 }
11 }
12
13 class A {
14 //調用到這個建構函式時:x=1,y=0
15 public A() {
16 //該函數在B類中被重載,因此輸出 x=1,y=0
17 Console.WriteLine("在A類的建構函式中");
18 PrintFields();
19 }
20 public virtual void PrintFields() { }
21 }
22 class B : A {
23 //衍生類別變數/靜態成員的執行順序優於基類
24 //先變數,因此,x=1,y=0
25 int x = 1;
26 int y;
27 //後建構函式,此時,先調用基類建構函式
28 public B() {
29 //運行到這裡時,A類建構函式已經執行完畢
30 y = -1;
31 Console.WriteLine("在B類的建構函式中");
32 //此時 x=1,y=-1
33 PrintFields();
34 }
35 public override void PrintFields() {
36 Console.WriteLine("x={0},y={1}", x, y);
37 }
38 }
39}
這裡備忘一下c#對象初始化順序及C++對象初始化順序:
C#對象初始化
1. 先變數後建構函式。變數先被初始化,然後建構函式被執行
2. 先靜態化後執行個體化。當一個類被訪問時,靜態變數和建構函式最先被初始化.接著是對象的執行個體化變數和建構函式被初始化
3. 先衍生類別後基類。對於變數和靜態建構函式,派生對象在基底物件之前被初始化.比如C類派生自B類,B類派生自A類,那麼變數和靜態建構函式被初始化次序是C-B-A.
4. 除了執行個體建構函式。對於執行個體建構函式,基類建構函式在衍生類別建構函式之前執行,執行個體建構函式被執行次序是A-B-C.
5. 不要假定變數的次序。Fields依據它們在源檔案中的聲明的順序依次初始化.然而,自從程式員和工具可以隨意安排變數的聲明後,你不應該在依靠變數任何特別的次序初始化
6. 對虛方法用兩個階段的構建。避免從一個構造器調用虛方法. 如果在初始化一個對象時需要調用一些虛方法,應在完整構造該對象的地方使用兩階段的構建,並隨後調用已構造對象的初始化方法。
C++建構函式調用順序
1. 如果類裡面有成員類,成員類的建構函式優先被調用;
2. 建立衍生類別的對象,基類的建構函式優先被調用(也優先於衍生類別裡的成員類);
3. 基類建構函式如果有多個基類,則建構函式的調用順序是某類在類派生表中出現的順序而不是它們在成員初始化表中的順序;
4. 成員類物件建構函數如果有多個成員類對象,則建構函式的調用順序是對象在類中被聲明的順序而不是它們出現在成員初始化表中的順序;
5. 衍生類別建構函式,作為一般規則衍生類別建構函式應該不能直接向一個基類資料成員賦值而是把值傳遞給適當的基類建構函式,否則兩個類的實現變成緊耦合的(tightly coupled)將更加難於正確地修改或擴充基類的實現。(基類設計者的責任是提供一組適當的基類建構函式)