若一个实例方法的声明中含有 virtual 修饰符,则称该方法为虚方法 (virtual method)。若其中没有 virtual 修饰符,则称该方法为非虚方法 (non-virtual method)。
在调用一个虚方法时,该调用所涉及的那个实例的运行时类型 (runtime type) 确定了要被调用的究竟是该方法的哪一个实现。在非虚方法调用中,实例的编译时类型 (compile-time type) 是决定性因素。
虚方法可以在派生类中重写 (override)。当某个实例方法声明包括 override 修饰符时,该方法将重写所继承的具有相同签名的虚方法。虚方法声明用于引入新方法,而重写方法声明则用于使现有的继承虚方法专用化(通过提供该方法的新实现)。
抽象 (abstract) 方法是没有实现的虚方法。抽象方法使用 abstract 修饰符进行声明,并且只有在同样被声明为 abstract 的类中才允许出现。抽象方法必须在每个非抽象派生类中重写。
下面的示例声明一个抽象类 Expression,它表示一个表达式树节点;它有三个派生类 Constant、VariableReference 和 Operation,它们分别实现了常量、变量引用和算术运算的表达式树节点。
Expression 实例的 Evaluate 方法将被调用,以计算给定的表达式的值,从而产生一个 double 值。该方法接受一个包含变量名称(作为哈希表项的键)和值(作为项的值)的 Hashtable 作为参数。Evaluate 方法是一个虚抽象方法,意味着非抽象派生类必须重写该方法以提供具体的实现。
Constant 的 Evaluate 实现只是返回所存储的常量。VariableReference 的实现在哈希表中查找变量名称,并返回产生的值。Operation 的实现先对左操作数和右操作数求值(通过递归调用它们的 Evaluate 方法),然后执行给定的算术运算。
下面的程序使用 Expression 类,对于不同的 x 和 y 值,计算表达式 x * (y + 2) 的值。
1 using System; 2 using System.Collections; 3 namespace Virtual_override_abstract_method 4 { 5 class Program 6 { 7 static void Main(string[] args) 8 { 9 Expression e = new Operation( 10 new VariableReference("x"), 11 '*', 12 new Operation( 13 new VariableReference("y"), 14 '+', 15 new Constant(2) 16 )); 17 Hashtable vars = new Hashtable(); 18 vars["x"] = 3; 19 vars["y"] = 5; 20 Console.WriteLine(e.Evaluate(vars)); // Outputs "21" 21 Console.Read(); 22 23 } 24 } 25 public abstract class Expression 26 { 27 public abstract double Evaluate(Hashtable vars); 28 } 29 public class Constant : Expression 30 { 31 double value; 32 public Constant(double value) 33 { 34 this.value = value; 35 } 36 public override double Evaluate(Hashtable vars) 37 { 38 return value; 39 } 40 } 41 public class VariableReference : Expression 42 { 43 string name; 44 public VariableReference(string name) 45 { 46 this.name = name; 47 } 48 public override double Evaluate(Hashtable vars) 49 { 50 object value = vars[name]; 51 if (value == null) 52 { 53 throw new Exception("Unknown variable: " + name); 54 } 55 return Convert.ToDouble(value); 56 } 57 } 58 public class Operation : Expression 59 { 60 Expression left; 61 char op; 62 Expression right; 63 public Operation(Expression left, char op, Expression right) 64 { 65 this.left = left; 66 this.op = op; 67 this.right = right; 68 } 69 public override double Evaluate(Hashtable vars) 70 { 71 double x = left.Evaluate(vars); 72 double y = right.Evaluate(vars); 73 switch (op) 74 { 75 case '+': return x + y; 76 case '-': return x - y; 77 case '*': return x * y; 78 case '/': return x / y; 79 } 80 throw new Exception("Unknown operator"); 81 } 82 } 83 }