5.2.2 改写方法 面向对象设计的重要原则就是多态性。不要理会高深的理论,多态性意味着:当基类程序员已设计好用于改写的方法时,在派生类中,你就可以重定义(改写)基类的方法。基类程序员可以用virtual 关键字设计方法: virtual void CanBOverridden() 当从基类派生时,所有你要做的就是在新方法中加入override关键字: override void CanBOverridden() 当改写一个基类的方法时,你必须明白,不能改变方法的访问属性——在这章的后面,你会学到更多关于访问修饰符的知识。 除了改写基类方法的事实外,还有另一个甚至更重要的改写特性。当把派生类强制转换成基类类型并接着调用虚拟方法时,被调用的是派生类的方法而不是基类的方法。 ((BaseClass)DerivedClassInstance).CanBOverridden(); 为了演示虚拟方法的概念,清单 5.4 显示如何创建一个三角形基类,它拥有一个可以被改写的成员方法(ComputeArea)。 清单 5.4 改写一个基类的方法 1: using System;2: 3: class Triangle4: {5: public virtual double ComputeArea(int a, int b, int c)6: {7: // Heronian formula8: double s = (a + b + c) / 2.0;9: double dArea = Math.Sqrt(s*(s-a)*(s-b)*(s-c));10: return dArea;11: }12: }13: 14: class RightAngledTriangle:Triangle15: {16: public override double ComputeArea(int a, int b, int c)17: { 18: double dArea = a*b/2.0;19: return dArea;20: }21: }22: 23: class TriangleTestApp24: {25: public static void Main()26: {27: Triangle tri = new Triangle();28: Console.WriteLine(tri.ComputeArea(2, 5, 6));29: 30: RightAngledTriangle rat = new RightAngledTriangle();31: Console.WriteLine(rat.ComputeArea(3, 4, 5));32: }33: } 基类Triangle定义了方法ComputeArea。它采用三个参数,返回一个double结果,且具有公共访问性。从Triangle类派生出的是RightAngledTriangle,它改写了ComputeArea 方法,并实现了自己的面积计算公式。两个类都被实例化,且在命名为TriangleTestApp的应用类的Main() 方法中得到验证。 我漏了解释第14行:class RightAngledTriangle : Triangle在类语句中冒号(:)表示RightAngledTriangle从类 Triangle派生。那就是你所必须要做的,以让C#知道你想把 Triangle当作RightAngledTriangle的基类。当仔细观察直角三角形的ComputeArea方法时,你会发现第3个参数并没有用于计算。但是,利用该参数就可以验证是否是“直角”。如清单5.5所示。 清单 5.5 调用基类实现 1: class RightAngledTriangle:Triangle2: {3: public override double ComputeArea(int a, int b, int c)4: {5: const double dEpsilon = 0.0001;6: double dArea = 0;7: if (Math.Abs((a*a + b*b - c*c)) > dEpsilon)8: {9: dArea = base.ComputeArea(a,b,c);10: }11: else12: {13: dArea = a*b/2.0;14: }15: 16: return dArea;17: }18: } 该检测简单地利用了毕达哥拉斯公式,对于直角三角形,检测结果必须为0。如果结果不为0,类就调用它基类的 ComputeArea来实现。 dArea = base.ComputeArea(a,b,c); 例子的要点为:通过显式地利用基类的资格检查,你就能轻而易举地调用基类实现改写方法。当你需要实现其在基类中的功能,而不愿意在改写方法中重复它时,这就非常有帮助。 5.2.3 方法屏蔽 重定义方法的一个不同手段就是要屏蔽基类的方法。当从别人提供的类派生类时,这个功能特别有价值。看清单 5.6,假设BaseClass由其他人所写,而你从它派生出 DerivedClass 。 清单 5.6 Derived Class 实现一个没有包含于 Base Class中的方法 1: using System;2: 3: class BaseClass4: {5: }6: 7: class DerivedClass:BaseClass8: {9: public void TestMethod()10: {11: Console.WriteLine("DerivedClass::TestMethod");12: }13: }14: 15: class TestApp16: {17: public static void Main()18: {19: DerivedClass test = new DerivedClass();20: test.TestMethod();21: }22: } 在这个例子中, DerivedClass 通过TestMethod()实现了一个额外的功能。但是,如果基类的开发者认为把TestMethod()放在基类中是个好主意,并使用相同的名字实现它时,会出现什么问题呢?(见清单5.7) 清单 5.7 Base Class 实现和 Derived Class相同的方法 1: class BaseClass2: {3: public void TestMethod()4: {5: Console.WriteLine("BaseClass::TestMethod");6: }7: }8: 9: class DerivedClass:BaseClass10: {11: public void TestMethod()12: {13: Console.WriteLine("DerivedClass::TestMethod");14: }15: } 在优秀的编程语言中,你现在会遇到一个真正的大麻烦。但是,C#会给你提出警告:hiding2.cs(13,14): warning CS0114: 'DerivedClass.TestMethod()' hides inherited member 'BaseClass.TestMethod()'. To make the current method override that implementation, add the override keyword. Otherwise add the new keyword.(hiding2.cs(13,14):警告 CS0114:'DerivedClass.TestMethod()' 屏蔽了所继承的成员 'BaseClass.TestMethod()'。要想使当前方法改写原来的实现,加上 override关键字。否则加上新的关键字。具有了修饰符new,你就可以告诉编译器,不必重写派生类或改变使用到派生类的代码,你的方法就能屏蔽新加入的基类方法。清单5.8 显示如何在例子中运用new修饰符。 清单 5.8 屏蔽基类方法 1: class BaseClass2: {3: public void TestMethod()4: {5: Console.WriteLine("BaseClass::TestMethod");6: }7: }8: 9: class DerivedClass:BaseClass10: {11: new public void TestMethod()12: {13: Console.WriteLine("DerivedClass::TestMethod");14: }15: }

评论