C# 构造函数中调用虚函数
using System; using System.Diagnostics; using System.Text; using System.Collections; using System.Collections.Generic; class Test { public class Base { public Base() { System.Console.WriteLine("Base.Base"); ABitDangerousCall(); } public virtual void ABitDangerousCall() { System.Console.WriteLine("Base.ABitDangerousCall"); } private class Inner { public Inner() { System.Console.WriteLine("Base.Inner.Inner"); } } private Inner inner = new Inner(); } class Derived : Base { public Derived() { System.Console.WriteLine("Derived.Derived"); ctorInitializedMember = 5; } // ctorInitializedMember is default initialized to zero before the constructor initializes it. private int ctorInitializedMember; private int derivedInt; public override void ABitDangerousCall() { System.Console.WriteLine(String.Format("Derived.ABitDangerousCallctorInitializedMember={0} derivedInt={1}", ctorInitializedMember, derivedInt)); } private class Inner { public Inner() { System.Console.WriteLine("Derived.Inner.Inner"); } } private Inner inner = new Inner(); } class Program { static void Main(string[] args) { Derived d = new Derived(); Console.Read(); } } }
输出结果:
Derived.Inner.Inner
Base.Inner.Inner
Base.Base
Derived.ABitDangerousCallctorInitializedMember=0 derivedInt=0
Derived.Derived
由于实例构造函数的调用顺序是先基类构造函数再子类构造函数,如上面例子,在基类中调用被子类重写了的虚函数ABitDangerousCall时,实际会调用子类的ABitDangerousCall,
该函数中使用了变量ctorInitializedMember,但此时子类的构造函数还没调用,故ctorInitializedMember未被赋值,这样就出现了不想看到的结果。我们本来是想想看到ctorInitializedMember=5的结果,结果却是0。
因此在基类的构造函数中调用虚函数是很危险的,项目大了代码复杂了就很容易造成极隐蔽的BUG。同时在子类的构造函数中调用虚函数也是要小心的,尽量避免,原因同上。
如果非要调用,那就要细心察看虚函数中的变量是否已经全部在前面处理妥当了
本文探讨了在C#中于构造函数内调用虚函数所带来的隐患,特别是当这些函数依赖于子类构造函数初始化的成员变量时。通过一个具体的代码示例,展示了这种做法可能导致意外的行为,并提醒开发者注意此类设计模式的潜在问题。
683

被折叠的 条评论
为什么被折叠?



