若一個實例方法的聲明中含有 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 }