• C#虚方法、重写方法和抽象方法


    若一个实例方法的声明中含有 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 }
  • 相关阅读:
    周五,远程连接及总体流程
    C++ 图片浏览
    深度解析Java内存的原型
    找不到class
    js读写cookie
    不利用临时变量,交换两个变量的值
    插入排序
    算法:一个排序(第一个最大,第二个最小,第三个其次大,第四其次小...)
    c#缓存介绍(1)
    JavaScript中创建自定义对象
  • 原文地址:https://www.cnblogs.com/zhangyongheng/p/4861186.html
Copyright © 2020-2023  润新知