5

好吧,最初我在每个派生类中有几个具有不同值的常量(如 MAX_SPEED)。这个想法是在基类的某些方法中使用这些值。那时我意识到我不能用常量做到这一点,所以我创建了只读属性。

我需要一种在实例化时将这些值分配给私有字段的方法,最好是在基类中。但首先我必须评估派生类中的原始值。由于这些是属性,我在定义时找不到初始化它们的方法,所以唯一的方法是在派生的构造函数中。

这就是问题所在:值在分配给基类中的私有字段后被初始化。我逃脱的解决方案是创建一个虚拟方法并在那里进行分配。

有没有办法从派生类调用基构造函数,以便首先调用派生构造函数的代码?

class BaseClass
{
    public BaseClass()
    {
        System.Console.WriteLine("This should be shown after");
    }
}

class DerivedClass : BaseClass
{
    public DerivedClass() : base()
    {
        System.Console.WriteLine("This should be shown first");
    }
}

当然,在示例中,它会以相反的方式工作。有解决办法吗?

4

3 回答 3

15

不,基类构造函数总是在派生类构造函数的主体之前执行。然而:

  • 派生类中的任何实例变量初始值设定项都在基类构造函数之前执行
  • 基类构造函数可以执行可以在派生类中重写的虚拟方法。不过,这几乎总是一个坏主意。(此时各种正常的前置条件都无效。您可以观察尚未设置的只读变量,因为它们将在构造函数主体中设置,例如。Ick。)

为了证明这两个:

using System;

class BaseClass
{
    public BaseClass()
    {
        VirtualMethod();
        Console.WriteLine("BaseClass ctor body");
    }

    public virtual void VirtualMethod()
    {
        Console.WriteLine("BaseClass.VirtualMethod");
    }
}

class DerivedClass : BaseClass
{
    int ignored = ExecuteSomeCode();

    public DerivedClass() : base()
    {
        Console.WriteLine("DerivedClass ctor body");
    }

    static int ExecuteSomeCode()
    {
        Console.WriteLine("Method called from initializer");
        return 5;
    }

    public override void VirtualMethod()
    {
        Console.WriteLine("DerivedClass.VirtualMethod");
    }
}

class Test
{
    static void Main()
    {
        new DerivedClass();
    }
}

输出:

Method called from initializer
DerivedClass.VirtualMethod
BaseClass ctor body
DerivedClass ctor body

此外,如果您的基类构造函数带有参数,那么您可以在派生类中执行一些代码以提供参数:

DerivedClass() : base(SomeStaticMethod())

不过,所有这些都相当臭。你的具体情况是什么?

于 2013-05-11T15:49:09.197 回答
8

不,你不能那样做。基类总是首先初始化。但是,您可以执行以下操作:

class BaseClass
{
    public BaseClass()
    {
        this.Initialize();
    }

    protected virtual void Initialize()
    {
        System.Console.WriteLine("This should be shown after");
    }
}

class DerivedClass : BaseClass
{
    public DerivedClass() : base()
    {
    }

    protected override void Initialize()
    {
        System.Console.WriteLine("This should be shown first");
        base.Initialize();
    }
}
于 2013-05-11T15:49:18.250 回答
0

另一种选择是将子类构造函数设置为 static ,以便它比父类构造函数首先执行。但它不宜违反 oop 设计

于 2016-09-21T12:53:02.807 回答