被修饰为虚拟的方法,实现是可变的,派生类可以重写基类的虚拟方法,也可以不重写基类的虚拟方法
publicclassSpace2


{3

4
publicstaticvoidMain(string[]args)5


{6

7
newOrder().Submit();//todefaultdepartment8
newShipOrder().Submit();//todefaultShipdepartment9
newInOrder().Submit();//todefaultdepartment10
newOutOrder().Submit();//todefaultOutdepartment/ntodefaultdepartment11
}12
}13

14

15
publicclassOrder16


{17

18
publicvirtualvoidSubmit()19


{20
System.Console.WriteLine("todefaultdepartment");21
}22
}23

24
publicclassShipOrder:Order25


{26
publicoverridevoidSubmit()27


{28
System.Console.WriteLine("todefaultShipdepartment");29
}30
}31

32
publicclassInOrder:Order33


{34

35
}36

37
publicclassOutOrder:Order38


{39
publicoverridevoidSubmit()40


{41
System.Console.WriteLine("todefaultOutdepartment");42
base.Submit();43
}44
}
在以上代码中,基类Order的Submit方法被修饰为virtual的,所以ShipOrder 和OutOrder 可以重新定义Submit的具体实现(用override 关键
字),并且还可以使用关键字base来调用基类的方法。
在讲到方法重写的时候,就必须要了解对象的数据类型转换。
在数据类型中,我们知道 数据容量小的(比如int)可以被安全的转换到数据容量大的(比如long)类型,反之是不安全的。
在对象世界正好相反,数据容量大的(子类)可以被安全的转换到数据容量小的(子类的基类)类型,反之是不安全的。
所以,所有的数据类型,都可以被安全的转为object,就是这个道理。
以下代码演示了对象的数据类型转换
Orderorder=newShipOrder();//可以,子类转为基类2
order.Submit();//todefaultShipdepartment3
System.Console.WriteLine("typeis{0}",order.GetType());//typeisShipOrder4
ShipOrdership=(ShipOrder)order;//可以,order是从ShipOrder转来的,但由于是基类转子类,必须强制类型转换5
order.Submit();//todefaultShipdepartment6
System.Console.WriteLine("typeis{0}",ship.GetType());//typeisShipOrder7
order=newOrder();8
ship=(ShipOrder)order;//不可以,order中没有ShipOrder的成员,编译正确,运行错误9
InOrderinorder=newInOrder();10
ship=(ShipOrder)inorder;//不可以,编译错误
在金庸小说《倚天屠龙记》中描述了virtual方法的含义
只听张三丰问道:“孩儿,你看清楚了没有?”张无忌道:“看清楚了。”张三丰道:“都记得了没有?”张无忌道:“已忘记了一小半。”
张三丰道:“好,那也难为了你。你自己去想想罢。”张无忌低头默想。过了一会,张三丰问道:“现下怎样了?”张无忌道:“已忘记了一
大半。”张三丰画剑成圈,问道:“孩儿,怎样啦?”张无忌道:“还有三招没忘记。”张三丰点点头,放剑归座。张无忌在殿上缓缓踱了一
个圈子,沉思半晌,又缓缓踱了半个圈子,抬起头来,满脸喜色,叫道:“这我可全忘了,忘得乾乾净净的了。”张三丰道:“不坏,不坏!
忘得真快”
publicclassShadowboxing2


{3
publicvirtualvoid白鹤亮翅()4


{5
}6
publicvirtualvoid如封似闭()7


{8
}9
publicvirtualvoid野马分鬃()10


{11
}12
}13

14
publicclassZhang:Shadowboxing//张无忌继承张三丰的拳法15


{16
publicoverridevoid白鹤亮翅()17


{18

19
}20

21
publicoverridevoid如封似闭()22


{23

24
}25

26
}
我们再来看抽象方法abstract
抽象的方法隐含的为虚拟的方法,抽象的方法要求派生类必须重写,抽象的方法没有方法体。
publicabstractclassOrder2


{3

4
publicabstractvoidSubmit();5
}6

7
publicclassShipOrder:Order8


{9
publicoverridevoidSubmit()10


{11
System.Console.WriteLine("todefaultShipdepartment");12
}13
}14

15
publicabstractclassInOrder:Order16


{17

18
}19

20
publicclassOutOrder:Order21


{22
publicoverridevoidSubmit()23


{24
System.Console.WriteLine("todefaultOutdepartment");25
}26
}
Order定义了抽象的Submit,因此只能定义返回值,方法名称,参数列表和访问修饰。有抽象方法的类一定是抽象类(抽象类不一定需要抽象方法)
子类可以实现抽象方法,比如ShipOrder和OutOrder
子类也可以不实现抽象方法,比如InOrder
金庸同志也同时描述了一个关于abstract的例子,威震天下的独孤九剑就是典型的抽象类型
独孤大侠是绝顶聪明之人,学他的剑法,要旨是在一个‘悟’字,决不在死记硬记。等到通晓了这九剑的剑意,则无所施而不可,便是将全部
变化尽数忘记,也不相干,临敌之际,更是忘记得越干净彻底,越不受原来剑法的拘束。
publicabstractclassAloneSwordsmanship//独孤九剑2


{3

4
publicabstractvoid破剑式();5
publicabstractvoid破刀式();6
publicabstractvoid破枪式();7
publicabstractvoid破鞭式();8
publicabstractvoid破索式();9
publicabstractvoid破掌式();10
publicabstractvoid破气式();11
}12

13
publicclassFeng:AloneSwordsmanship//风清扬继承独孤九剑14


{15

16
publicoverridevoid破剑式()17


{18
}19

20
publicoverridevoid破刀式()21


{22
}23

24
publicoverridevoid破枪式()25


{26
}27

28
publicoverridevoid破鞭式()29


{30
}31

32
publicoverridevoid破索式()33


{34
}35

36
publicoverridevoid破掌式()37


{38
}39

40
publicoverridevoid破气式()41


{42
}43
}44

45
publicclassChong:Feng//令狐冲继承风清扬46


{47

48
}
重写方法override
只可以重写基类为虚拟或抽象的方法
必须有相同的签名、不可以更改访问修饰、不可以更改返回值、不可以更改参数列表、不可以更改方法名称、被重写后的方法一定可以再重写
不能是静态方法static、不能是密封方法
所以严格的来讲,令狐冲从风清扬那里得到的是已经重写后的类,不是抽象方法了,风清扬从独孤九剑那里获取的才是真正的抽象剑法。
当基类中没有定义为虚拟或者抽象的方法时
一般的方法是不允许被override的。不过有时候子类一定要有自己的特色,那怎么办呢?new一个方法。
publicclassOutOrder:Order2


{3
publicoverridevoidSubmit()4


{5
System.Console.WriteLine("todefaultOutdepartment");6
}7

8
publicvoidSend()9


{10
System.Console.WriteLine("SendtoOutdepartment");11
}12
}13

14
publicclassDelivery:OutOrder15


{16

17
publicnewvoidSend()18


{19
System.Console.WriteLine("SendtoDeliverydepartment");20
}21
}
注意OutOrder 定义的Send方法没有可以重写的修饰,但子类Delivery 通过new void Send覆盖了基类的方法,因此请注意下面的调用代码
OutOrderoutorder=newOutOrder();2
outorder.Send();//SendtoOutdepartment3
Deliverydelivery=newDelivery();4
delivery.Send();//SendtoDeliverydepartment5
outorder=delivery;6
outorder.Send();//注意:SendtoOutdepartment
当子类用new覆盖了基类的方法时,若引用变量为子类,则运行的是子类的new方法;若引用变量是基类,则运行基类方法。
刚才说到了抽象方法和抽象类,我们要再次提醒,有抽象方法的一定是抽象类,但是抽象类不一定包含抽象方法。
嫁衣神功就是典型的没有抽象方法的抽象类,嫁衣神功的修练者本身不能享用嫁衣神功的威力(类不能实例化),要把功力传给他人(必须有子类继承)
以下是一个没有抽象方法的抽象类的例子:
publicabstractclassContract//合同

{
/*publicContract()
{
BillCount++;
}
*/
//抽象类没有必要写构造函数
publicstringBuyer;//买方
publicstringSeller;//卖方
protecteddoubleCostPrice;//成本价
publicdoubleSalePrice;//销售价格
publicboolCheckPrice()

{
returnSalePrice>CostPrice;
}
}
publicclassSaleContract:Contract

{
}

被折叠的 条评论
为什么被折叠?



