C# 多態(tài)性

c# 多態(tài)性

多態(tài)是同一個行為具有多個不同表現(xiàn)形式或形態(tài)的能力。

多態(tài)性意味著有多重形式。在面向?qū)ο缶幊谭妒街?,多態(tài)性往往表現(xiàn)為"一個接口,多個功能"。

多態(tài)性可以是靜態(tài)的或動態(tài)的。在靜態(tài)多態(tài)性中,函數(shù)的響應(yīng)是在編譯時發(fā)生的。在動態(tài)多態(tài)性中,函數(shù)的響應(yīng)是在運行時發(fā)生的。

在 c# 中,每個類型都是多態(tài)的,因為包括用戶定義類型在內(nèi)的所有類型都繼承自 object。

多態(tài)就是同一個接口,使用不同的范例而執(zhí)行不同操作,如圖所示:

現(xiàn)實中,比如我們按下 f1 鍵這個動作:

  • 如果當前在 flash 界面下彈出的就是 as 3 的幫助文檔;
  • 如果當前在 word 下彈出的就是 word 幫助;
  • 在 windows 下彈出的就是 windows 幫助和支持。

同一個事件發(fā)生在不同的對象上會產(chǎn)生不同的結(jié)果。

 

1. 靜態(tài)多態(tài)性

在編譯時,函數(shù)和對象的連接機制被稱為早期綁定,也被稱為靜態(tài)綁定。c# 提供了兩種技術(shù)來實現(xiàn)靜態(tài)多態(tài)性。分別為:

  • 函數(shù)重載
  • 運算符重載

運算符重載將在下一章節(jié)討論,接下來我們將討論函數(shù)重載。

 

2. 函數(shù)重載

您可以在同一個范圍內(nèi)對相同的函數(shù)名有多個定義。函數(shù)的定義必須彼此不同,可以是參數(shù)列表中的參數(shù)類型不同,也可以是參數(shù)個數(shù)不同。不能重載只有返回類型不同的函數(shù)聲明。

下面的范例演示了幾個相同的函數(shù) add(),用于對不同個數(shù)參數(shù)進行相加處理:

using system;
namespace polymorphismapplication
{
? ? public class testdata ?
? ? { ?
? ? ? ? public int add(int a, int b, int c) ?
? ? ? ? { ?
? ? ? ? ? ? return a + b + c; ?
? ? ? ? } ?
? ? ? ? public int add(int a, int b) ?
? ? ? ? { ?
? ? ? ? ? ? return a + b; ?
? ? ? ? } ?
? ? } ?
? ? class program ?
? ? { ?
? ? ? ? static void main(string[] args) ?
? ? ? ? { ?
? ? ? ? ? ? testdata dataclass = new testdata();
? ? ? ? ? ? int add1 = dataclass.add(1, 2); ?
? ? ? ? ? ? int add2 = dataclass.add(1, 2, 3);

? ? ? ? ? ? console.writeline("add1 :" + add1);
? ? ? ? ? ? console.writeline("add2 :" + add2); ?
? ? ? ? } ?
? ? } ?
}

下面的范例演示了幾個相同的函數(shù) print(),用于打印不同的數(shù)據(jù)類型:

using system;
namespace polymorphismapplication
{
? ?class printdata
? ?{
? ? ? void print(int i)
? ? ? {
? ? ? ? ?console.writeline("輸出整型: {0}", i );
? ? ? }

? ? ? void print(double f)
? ? ? {
? ? ? ? ?console.writeline("輸出浮點型: {0}" , f);
? ? ? }

? ? ? void print(string s)
? ? ? {
? ? ? ? ?console.writeline("輸出字符串: {0}", s);
? ? ? }
? ? ? static void main(string[] args)
? ? ? {
? ? ? ? ?printdata p = new printdata();
? ? ? ? ?// 調(diào)用 print 來打印整數(shù)
? ? ? ? ?p.print(1);
? ? ? ? ?// 調(diào)用 print 來打印浮點數(shù)
? ? ? ? ?p.print(1.23);
? ? ? ? ?// 調(diào)用 print 來打印字符串
? ? ? ? ?p.print("hello yapf");
? ? ? ? ?console.readkey();
? ? ? }
? ?}
}

當上面的代碼被編譯和執(zhí)行時,它會產(chǎn)生下列結(jié)果:

輸出整型: 1
輸出浮點型: 1.23
輸出字符串: hello yapf

 

3. 動態(tài)多態(tài)性

c# 允許您使用關(guān)鍵字 abstract 創(chuàng)建抽象類,用于提供接口的部分類的實現(xiàn)。當一個派生類繼承自該抽象類時,實現(xiàn)即完成。抽象類包含抽象方法,抽象方法可被派生類實現(xiàn)。派生類具有更專業(yè)的功能。

請注意,下面是有關(guān)抽象類的一些規(guī)則:

  • 您不能創(chuàng)建一個抽象類的范例。
  • 您不能在一個抽象類外部聲明一個抽象方法。
  • 通過在類定義前面放置關(guān)鍵字 sealed,可以將類聲明為密封類。當一個類被聲明為 sealed 時,它不能被繼承。抽象類不能被聲明為 sealed。

下面的程序演示了一個抽象類:

using system;
namespace polymorphismapplication
{
? ?abstract class shape
? ?{
? ? ? ?abstract public int area();
? ?}
? ?class rectangle: ?shape
? ?{
? ? ? private int length;
? ? ? private int width;
? ? ? public rectangle( int a=0, int b=0)
? ? ? {
? ? ? ? ?length = a;
? ? ? ? ?width = b;
? ? ? }
? ? ? public override int area ()
? ? ? { 
? ? ? ? ?console.writeline("rectangle 類的面積:");
? ? ? ? ?return (width * length); 
? ? ? }
? ?}

? ?class rectangletester
? ?{
? ? ? static void main(string[] args)
? ? ? {
? ? ? ? ?rectangle r = new rectangle(10, 7);
? ? ? ? ?double a = r.area();
? ? ? ? ?console.writeline("面積: {0}",a);
? ? ? ? ?console.readkey();
? ? ? }
? ?}
}

當上面的代碼被編譯和執(zhí)行時,它會產(chǎn)生下列結(jié)果:

rectangle 類的面積:
面積: 70

當有一個定義在類中的函數(shù)需要在繼承類中實現(xiàn)時,可以使用虛方法

虛方法是使用關(guān)鍵字 virtual 聲明的。

虛方法可以在不同的繼承類中有不同的實現(xiàn)。

對虛方法的調(diào)用是在運行時發(fā)生的。

動態(tài)多態(tài)性是通過 抽象類虛方法 實現(xiàn)的。

以下范例創(chuàng)建了 shape 基類,并創(chuàng)建派生類 circle、 rectangle、triangle, shape 類提供一個名為 draw 的虛擬方法,在每個派生類中重寫該方法以繪制該類的指定形狀。

using system;
using system.collections.generic;

public class shape
{
? ? public int x { get; private set; }
? ? public int y { get; private set; }
? ? public int height { get; set; }
? ? public int width { get; set; }
? ?
? ? // 虛方法
? ? public virtual void draw()
? ? {
? ? ? ? console.writeline("執(zhí)行基類的畫圖任務(wù)");
? ? }
}

class circle : shape
{
? ? public override void draw()
? ? {
? ? ? ? console.writeline("畫一個圓形");
? ? ? ? base.draw();
? ? }
}
class rectangle : shape
{
? ? public override void draw()
? ? {
? ? ? ? console.writeline("畫一個長方形");
? ? ? ? base.draw();
? ? }
}
class triangle : shape
{
? ? public override void draw()
? ? {
? ? ? ? console.writeline("畫一個三角形");
? ? ? ? base.draw();
? ? }
}

class program
{
? ? static void main(string[] args)
? ? {
? ? ? ? // 創(chuàng)建一個 list<shape> 對象,并向該對象添加 circle、triangle 和 rectangle
? ? ? ? var shapes = new list<shape>
? ? ? ? {
? ? ? ? ? ? new rectangle(),
? ? ? ? ? ? new triangle(),
? ? ? ? ? ? new circle()
? ? ? ? };

? ? ? ? // 使用 foreach 循環(huán)對該列表的派生類進行循環(huán)訪問,并對其中的每個 shape 對象調(diào)用 draw 方法 
? ? ? ? foreach (var shape in shapes)
? ? ? ? {
? ? ? ? ? ? shape.draw();
? ? ? ? }

? ? ? ? console.writeline("按下任意鍵退出。");
? ? ? ? console.readkey();
? ? }

}

當上面的代碼被編譯和執(zhí)行時,它會產(chǎn)生下列結(jié)果:

畫一個長方形
執(zhí)行基類的畫圖任務(wù)
畫一個三角形
執(zhí)行基類的畫圖任務(wù)
畫一個圓形
執(zhí)行基類的畫圖任務(wù)
按下任意鍵退出。

下面的程序演示通過虛方法 area() 來計算不同形狀圖像的面積:

using system;
namespace polymorphismapplication
{
? ?class shape 
? ?{
? ? ? protected int width, height;
? ? ? public shape( int a=0, int b=0)
? ? ? {
? ? ? ? ?width = a;
? ? ? ? ?height = b;
? ? ? }
? ? ? public virtual int area()
? ? ? {
? ? ? ? ?console.writeline("父類的面積:");
? ? ? ? ?return 0;
? ? ? }
? ?}
? ?class rectangle: shape
? ?{
? ? ? public rectangle( int a=0, int b=0): base(a, b)
? ? ? {

? ? ? }
? ? ? public override int area ()
? ? ? {
? ? ? ? ?console.writeline("rectangle 類的面積:");
? ? ? ? ?return (width * height); 
? ? ? }
? ?}
? ?class triangle: shape
? ?{
? ? ? public triangle(int a = 0, int b = 0): base(a, b)
? ? ? {
? ? ? 
? ? ? }
? ? ? public override int area()
? ? ? {
? ? ? ? ?console.writeline("triangle 類的面積:");
? ? ? ? ?return (width * height / 2); 
? ? ? }
? ?}
? ?class caller
? ?{
? ? ? public void callarea(shape sh)
? ? ? {
? ? ? ? ?int a;
? ? ? ? ?a = sh.area();
? ? ? ? ?console.writeline("面積: {0}", a);
? ? ? }
? ?} ?
? ?class tester
? ?{
? ? ? 
? ? ? static void main(string[] args)
? ? ? {
? ? ? ? ?caller c = new caller();
? ? ? ? ?rectangle r = new rectangle(10, 7);
? ? ? ? ?triangle t = new triangle(10, 5);
? ? ? ? ?c.callarea(r);
? ? ? ? ?c.callarea(t);
? ? ? ? ?console.readkey();
? ? ? }
? ?}
}

當上面的代碼被編譯和執(zhí)行時,它會產(chǎn)生下列結(jié)果:

rectangle 類的面積:
面積:70
triangle 類的面積:
面積:25

下一節(jié):c# 運算符重載

c# 教程

相關(guān)文章
亚洲国产精品第一区二区,久久免费视频77,99V久久综合狠狠综合久久,国产免费久久九九免费视频