C 解惑 1 在建構函式內呼叫虛方法

2022-01-30 23:16:18 字數 2782 閱讀 9333

在c#中,用virtual關鍵字修飾的方法(屬性、事件)稱為虛方法(屬性、事件),表示該方法可以由派生類重寫(override)。虛方法是.net中的重要概念,可以說在某種程度上,虛方法使得多型成為可能。

然而虛方法的使用卻存在著很大學問,如果濫用的話勢必對程式產生很大的負面影響。比如下面這個例子:

public class puzzle

public virtual string name

public virtual void solve()

}

如果您的visual studio沒有安裝resharper,那麼上面的**不會有任何異常。但如果安裝了,在建構函式內部給name賦值和呼叫solve時就會在下面產生乙個波浪線,即警告:virtual member call in constructor。

這是什麼原因呢?我們在建構函式中呼叫虛方法,礙著resharper什麼事兒了?

其實這個警告就是提醒我們不要在非封閉型別的建構函式內呼叫虛方法或虛屬性。但為什麼這樣做不合適呢?在解惑之前,我們先來了解兩個概念。

我們先來看這樣一段**:

class base

}class derived : base

}static class program

}

猜一猜它的輸出結果是什麼?

你也許已經猜到了,它的結果是:

base constructor

derived constructor

我們在初始化乙個物件時,總是會先執行基類的建構函式,然後再執行子類的建構函式

我們再來看一段**:

class base

public virtual void v()

}class derived : base

public override void v()

}static class program

}

再來猜一猜輸出結果吧。

貌似應該是:

base.m

base.v

derived.m

derived.v

但執行一下會發現,真正的結果是這樣的:

base.m

derived.v

derived.m

derived.v

這是為什麼呢?

原來對於非虛方法呼叫,編譯器會進行一些額外的「動作」。比如找出所呼叫物件的實際型別,以訪問正確的方法表(呼叫b.v()的時候就會找到變數b的實際型別derived,從而輸出derived.v)。

現在回到我們最初的謎題,virtual member call in constructor。結合以上兩個知識點,會有哪些發現?

我們稍微改造一下虛方法呼叫的那個例子。

class foo

public void bar()

}class base

public virtual void v()

}class derived : base

public override void v()

}

base的建構函式中呼叫虛方法v()時,resharper會給出virtual member call in constructor的警告。這是因為v可以在base的任意子類中被改寫(override),而這種改寫,很有可能使得它依賴於自己的建構函式,如上例所示。而由於之前提到的型別初始化順序,在執行base b = new derived();這樣的**時,base的建構函式要早於derived的建構函式執行,因此在執行到foo.bar()foo還是個空引用。

明白了嗎?我們來簡單總結一下。virtual member call in constructor的警告是因為,對於base b = new derived();這樣的**:

基類建構函式的執行要早於子類建構函式

基類建構函式中對於虛方法的呼叫,實際呼叫的是子類中重寫的虛方法

因此,resharper會警告我們,這麼做存在隱患。

我們能完全避免這麼做嗎?很遺憾,答案是不能。比如如果專案中使用了nhibernate,框架本身要求orm實體類中,所有與資料庫列具有對應關係的屬性都必須為虛屬性。這是因為nhibernate為了實現延遲載入,會為每個實體類生成proxy,這些proxy需要重寫實體類中屬性的getter/setter。而有些時候,為了業務需要,我們不得不在實體類的建構函式中對這些屬性進行某些操作(比如初始化)。

我認為這麼做是技術選型所致的必然結果,是完全可以接受的。但我們要注意,在**中保證那些可能會被繼承的實體,在子類中重寫那些虛屬性時,不要依賴於子類自身的建構函式(這幾乎是可以保證的,因為與資料庫列對映的屬性,只能是最簡單的getter/setter)。

在建構函式內呼叫non final函式時要當心

當程式新建乙個class物件時,class建構函式會被呼叫。建構函式的目的在於將物件初始化。建構函式在執行期間可以呼叫class的某些函式,這很普遍,因為那些被呼叫的函式或許包含一些初始化動作。舉個例子 class base public int lookup public int value cl...

C 建構函式中呼叫虛函式

我們知道 c 中的多型使得可以根據物件的真實型別 動態型別 呼叫不同的虛函式。這種呼叫都是物件已經構建完成的情況。那如果在建構函式中呼叫虛函式,會怎麼樣呢?有這麼一段 class a virtual void func void test public int m ival class b publ...

C 建構函式中呼叫虛函式

談談關於建構函式中呼叫虛函式的情況,僅討論單繼承,不考慮虛擬繼承和多重繼承。測試平台 vs2013 win7x64 乙個例子 include include class base public virtual void func class deri public base public virtu...