c# - 为属性赋值时在构造函数中调用虚拟成员

标签 c# properties resharper virtual abstract-class

我有一个抽象类和一个派生类。抽象类定义了一个名为 Message 的抽象属性。在派生类中,属性是通过覆盖抽象属性来实现的。派生类的构造函数采用字符串参数并将其分配给其 Message 属性。在 Resharper 中,此分配会导致警告“构造函数中的虚拟成员调用”。

AbstractClass 有这样的定义:

public abstract class AbstractClass {
    public abstract string Message { get; set; }

    protected AbstractClass() {}

    public abstract void PrintMessage();
}

DerivedClass如下:

using System;

public class DerivedClass : AbstractClass {
    private string _message;

    public override string Message {
        get { return _message; }
        set { _message = value; }
    }

    public DerivedClass(string message) {
        Message = message; // Warning: Virtual member call in a constructor
    }

    public DerivedClass() : this("Default DerivedClass message") {}

    public override void PrintMessage() {
        Console.WriteLine("DerivedClass PrintMessage(): " + Message);
    }
}

我确实发现了一些关于此警告的其他问题,但在那些情况下,确实存在对方法的调用。例如,在 this question 中,Matt Howels 的回答包含一些示例代码。为了便于引用,我会在这里重复一遍。

class Parent {
    public Parent() {
        DoSomething();
    }
    protected virtual void DoSomething() {};
}

class Child : Parent {
    private string foo;
    public Child() { foo = "HELLO"; }
    protected override void DoSomething() {
        Console.WriteLine(foo.ToLower());
    }
}

Matt 没有描述警告会出现什么错误,但我假设它会在调用 Parent 构造函数中的 DoSomething 时出现。在这个例子中,我理解了被调用的虚拟成员的含义。成员调用发生在基类中,基类中只存在一个虚方法。

然而,在我的情况下,我不明白为什么为 Message 赋值会调用虚拟成员。 Message 属性的调用和实现都在派生类中定义。

虽然我可以通过使我的派生类密封来消除错误,但我想了解为什么这种情况会导致警告。

更新 根据 Brett 的回答,我尽力创建了一个派生自 DerivedClass 的 ChildClass,它最终会导致异常。这是我想出的:

using System;

public class ChildClass : DerivedClass {
    private readonly string _foo;

    public ChildClass() : base("Default ChildClass Message") {
        _foo = "ChildClass foo";
    }

    public override string Message {
        get { return base.Message; }
        set {
            base.Message = value;
            Console.WriteLine(_foo.ToUpper() + " received " + value);
        }
    }
}

当然,在 Message setter 中使用 _foo 有点傻,但重点是 ReSharper 没有发现此类有任何问题。

但是,如果您尝试在这样的程序中使用 ChildClass:

internal class Program {
    private static void Main() {
        var childClass = new ChildClass();
        childClass.PrintMessage();
    }
}

创建 ChildClass 对象时,您将得到一个 NullReferenceException。 ChildClass 尝试使用 _foo.ToUpper() 时将引发异常,因为 _foo 尚未初始化。

最佳答案

这是因为您的 Message 属性可以被 class ChildClass : DerivedClass 覆盖 - 此时可以从 中的 ctor 调用 Message on ChildClass 中的代码DerivedClass,您的 ChildClass 实例可能未完全初始化。

这就是为什么让你的 DerivedClass 密封解决问题 - 它不能被继承。

关于c# - 为属性赋值时在构造函数中调用虚拟成员,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4625614/

相关文章:

c# - 父类和子类中的方法名称相同?

c# - .NET 框架中的 Span<T> 和流

c# - 获取/设置 RichTextBox 的第一个可见行

java - 从 Properties 实例获取使用的文件名

c# - Resharper 单元测试未运行

visual-studio-2012 - 为什么我在运行我的项目后失去了 JavaScript 智能感知?

c# - .net 网页浏览器控件

java - Java 系统属性存储在 Mac 上的什么位置?

javascript - 关于 Javascript 中函数对象作用域的快速介绍(我希望很简单!)...ode>

c# - Resharper 6.1 VS 2010 - 无法编辑只读文件。 (在内存中)