8

我一直在用 RhinoMocks 进行一些模拟,它要求将模拟的方法设为虚拟。这很好,除非我们有一个自定义框架,其中包含我要模拟的方法,这些方法当前未标记为虚拟。

我无法预见将这些方法设为虚拟会有任何问题,但我想知道将方法设为虚拟有哪些潜在危险需要注意?

4

3 回答 3

9

实际上,如果该方法没有被设计为被覆盖并且有人覆盖它,那么它可能会非常有问题。特别是,永远不要从构造函数中调用虚方法。考虑:

class Base {
    public Base() {
       InitializeComponent();
    }
    protected virtual void InitializeComponent() {
        ...
    }
}

class Derived : Base {
    private Button button1;
    public Derived() : base() {
        button1 = new Button();
    }
    protected override void InitializeComponent() {
        button1.Text = "I'm gonna throw a null reference exception"
    }
}

Derived 类可能不知道虚方法调用将导致其 InitializeComponent 方法在其自己的构造函数的单行运行之前被调用。

于 2008-09-26T01:51:04.123 回答
4
  • 如果您的用户覆盖了您的虚拟方法,您将无法在不破坏代码的情况下再次密封它们。
  • 您从构造函数调用的任何虚拟方法都可能归结为派生实现,如果它们不调用基方法并且构造函数依赖于它,则对象可能处于无效状态
于 2008-09-26T01:33:09.720 回答
2

Ayende 对虚拟方法的工作方式有很好的处理:

http://ayende.com/Blog/archive/2007/01/05/HowVirtualMethodsWork.aspx

于 2008-09-26T01:35:57.530 回答