在C#的學習中,容易混淆virtual方法和abstract方法的使用,現在來討論一下二者的區別。二者都牽涉到在派生類中與override的配合使用。
一、Virtual方法(虛方法)
virtual 關鍵字用於在基類中修飾方法。virtual的使用會有兩種情況:
情況1:在基類中定義了virtual方法,但在派生類中沒有重寫該虛方法。那麼在對派生類實例的調用中,該虛方法使用的是基類定義的方法。
情況2:在基類中定義了virtual方法,然後在派生類中使用override重寫該方法。那麼在對派生類實例的調用中,該虛方法使用的是派生重寫的方法。
二、Abstract方法(抽象方法)
abstract關鍵字只能用在抽象類中修飾方法,並且沒有具體的實現。抽象方法的實現必須在派生類中使用override關鍵字來實現。
接口和抽象類:
最本質的區別:抽象類是一個不完全的類,是對對象的抽象,而接口是一種行為規范。
C# 是面向對象的程序設計語言,每一個函數都屬於一個類。
Static:當一個方法被聲明為Static時,這個方法是一個靜態方法,編譯器會在編譯時保留這個方法的實現。也就是說,這個方法屬於類,但是不屬於任何成員,不管這個類的實例是否存在,它們都會存在。就像入口函數Static void Main,因為它是靜態函數,所以可以直接被調用。
Virtua:當一個方法被聲明為Virtual時,它是一個虛擬方法,直到你使用ClassName variable = new ClassName();聲明一個類的實例之前,它都不存在於真實的內存空間中。這個關鍵字在類的繼承中非常常用,用來提供類方法的多態性支持。
overrride:表示重寫 這個類是繼承於Shape類
public override double Area 這個屬性再shape中肯定存在 但是這裡我們不想用shape中的 所以要重寫
virtual,abstract是告訴其它想繼承於他的類 你可以重寫我的這個方法或屬性,否則不允許。
一個生動的例子 :老爸表示基類(被繼承的類) 兒子表示子類(繼承的類)
老爸用virtual告訴兒子:"孩子,你要繼承我的事業,在這塊上面可以自己繼續發展你自己的"
兒子用override告訴全世界:"這個我可不是直接拿我爸的,他只是指個路給我,是我自己奮斗出來的"
abstract:抽象方法聲明使用,是必須被派生類覆寫的方法,抽象類就是用來被繼承的;可以看成是沒有實現體的虛方法;如果類中包含抽象方法,那麼類就必須定義為抽象類,不論是否還包含其他一般方法;抽象類不能有實體的。
實例解答:
interface:用來聲明接口
1.只提供一些方法規約,不提供方法主體. 如:
public interface IPerson
{
void getName();//不包含方法主體
}
2.方法不能用public abstract等修飾,無字段變量,無構造函數。
3.方法可包含參數。 如
public interface IPerson
{
void getAge(string s);
}
一個例子(例1):
public interface IPerson
{
IPerson(); //錯誤
string name; //錯誤
public void getIDcard();//錯誤
void getName(); //right
void getAge(string s); //right
}
實現interface的類
1.與繼承類的格式一致,如 public class Chinese:IPerson{}
2.必須實現 interface 中的各個方法
例2,繼承例1
public class Chinese:IPerson
{
public Chinese(){} //添加構造
public void getName(){} //實現getName()
public void getAge(string s){} //實現getAge()
}
abstract:聲明抽象類、抽象方法
1.抽象方法所在類必須為抽象類
2.抽象類不能直接實例化,必須由其派生類實現。
3.抽象方法不包含方法主體,必須由派生類以override方式實現此方法,這點跟interface中的方法類似
如
public abstract class Book
{
public Book()
{
}
public abstract void getPrice(); //抽象方法,不含主體
public virtual void getName() //虛方法,可覆蓋
{
Console.WriteLine("this is a test:virtual getName()");
}
public virtual void getContent() //虛方法,可覆蓋
{
Console.WriteLine("this is a test:virtual getContent()");
}
public void getDate() //一般方法,若在派生類中重寫,須使用new關鍵字
{
Console.WriteLine("this is a test: void getDate()");
}
}
public class JavaBook:Book
{
public override void getPrice() //實現抽象方法,必須實現
{
Console.WriteLine("this is a test:JavaBook override abstract getPrice()");
}
public override void getName() //覆蓋原方法,不是必須的
{
Console.WriteLine("this is a test:JavaBook override virtual getName()");
}
}
測試如下:
public class test
{
public test()
{
JavaBook jbook=new JavaBook();
jbook.getPrice(); //將調用JavaBook中getPrice()
jbook.getName(); //將調用JavaBook中getName()
jbook.getContent(); //將調用Book中getContent()
jbook.getDate(); //將調用Book中getDate()
}
public static void Main()
{
test t=new test();
}
}
virtual:標記方法為虛方法
1.可在派生類中以override覆蓋此方法
2.不覆蓋也可由對象調用
3.無此標記的方法(也無其他標記),重寫時需用new隱藏原方法
abstract 與virtual : 方法重寫時都使用 override 關鍵字
接口定義以大寫字母I開頭。方法只定義其名稱,在C#中,方法默認是公有方法;用public修飾方法是不允許的,否則會出現編譯錯誤;接口可以從別的接口繼承,如果是繼承多個接口,則父接口列表用逗號間隔。
接口可以通過類來實現,當類的基列表同時包含基類和接口時,列表中首先出現的是基類;類必須要實現其抽象方法;
接口使用:見代碼(轉)
interface使用
interface使用(實例一)
using System;
namespace Dage.Interface
{
//打印機接口
public interface IPrint
{
string returnPrintName();
}
}
//--------------------------------------------
using System;
using Dage.Interface;
namespace Dage.Print
{
//HP牌打印機類
public class HP: IPrint
{
public string returnPrintName()
{
return "這是HP牌打印機";
}
}
}
//--------------------------------------------
using System;
namespace Dage.Print
{
//Eps牌打印機類
public class Eps: IPrint
{
public string returnPrintName()
{
return "這是Eps牌打印機";
}
}
}
//--------------------------------------------
using System;
using Dage.Interface;
namespace Dage
{
//打印類
public class Printer
{
public Printer()
{}
public string PrintName(IPrint iPrint)
{
return iPrint.returnPrintName();
}
}
}
//--------------------------------------------
--WinFrom中調用代碼:
private void button1_Click(object sender, System.EventArgs e)
{
Printer p= new Printer();
switch (this.comboBox1.Text)
{
case "HP":
MessageBox.Show(p.PrintName(new HP()));
break;
case "Eps":
MessageBox.Show(p.PrintName(new Eps()));
break;
default:
MessageBox.Show("沒有發現這個品牌!");
break;
}
}