我一直在用 RhinoMocks 进行一些模拟,它要求将模拟的方法设为虚拟。这很好,除非我们有一个自定义框架,其中包含我要模拟的方法,这些方法当前未标记为虚拟。
我无法预见将这些方法设为虚拟会有任何问题,但我想知道将方法设为虚拟有哪些潜在危险需要注意?
我一直在用 RhinoMocks 进行一些模拟,它要求将模拟的方法设为虚拟。这很好,除非我们有一个自定义框架,其中包含我要模拟的方法,这些方法当前未标记为虚拟。
我无法预见将这些方法设为虚拟会有任何问题,但我想知道将方法设为虚拟有哪些潜在危险需要注意?
实际上,如果该方法没有被设计为被覆盖并且有人覆盖它,那么它可能会非常有问题。特别是,永远不要从构造函数中调用虚方法。考虑:
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 方法在其自己的构造函数的单行运行之前被调用。
Ayende 对虚拟方法的工作方式有很好的处理:
http://ayende.com/Blog/archive/2007/01/05/HowVirtualMethodsWork.aspx