在C#的學習中,容易混淆virtual方法和abstract方法的使用,如今來討論一下兩者的區別。兩者都牽涉到在派生類中與override的配合使用。
1、Virtual方法(虛方法)
virtual 關鍵字用於在基類中修飾方法。virtual的使用會有兩種狀況:
狀況1:在基類中定義了virtual方法,但在派生類中沒有重寫該虛方法。那麼在對派生類實例的調用中,該虛方法使用的是基類定義的方法。
狀況2:在基類中定義了virtual方法,而後在派生類中使用override重寫該方法。那麼在對派生類實例的調用中,該虛方法使用的是派生重寫的方法。
2、Abstract方法(抽象方法)
abstract關鍵字只能用在抽象類中修飾方法,而且沒有具體的實現。抽象方法的實現必須在派生類中使用override關鍵字來實現。
接口和抽象類:ide
最本質的區別:抽象類是一個不徹底的類,是對對象的抽象,而接口是一種行爲規範。函數
C# 是面向對象的程序設計語言,每個函數都屬於一個類。 Static:當一個方法被聲明爲Static時,這個方法是一個靜態方法,編譯器會在編譯時保留這個方法的實現。也就是說,這個方法屬於類,可是不屬於任何成員,無論這個類的實例是否存在,它們都會存在。就像入口函數Static void Main,由於它是靜態函數,因此能夠直接被調用。學習
Virtua:當一個方法被聲明爲Virtual時,它是一個虛擬方法,直到你使用ClassName variable = new ClassName();聲明一個類的實例以前,它都不存在於真實的內存空間中。這個關鍵字在類的繼承中很是經常使用,用來提供類方法的多態性支持。 測試
overrride:表示重寫 這個類是繼承於Shape類 public override double Area 這個屬性再shape中確定存在 可是這裏咱們不想用shape中的 因此要重寫 virtual,abstract是告訴其它想繼承於他的類 你能夠重寫個人這個方法或屬性,不然不容許。this
一個生動的例子 :老爸表示基類(被繼承的類) 兒子表示子類(繼承的類) 老爸用virtual告訴兒子:"孩子,你要繼承個人事業,在這塊上面能夠本身繼續發展你本身的" 兒子用override告訴全世界:"這個我可不是直接拿我爸的,他只是指個路給我,是我本身奮鬥出來的"
abstract:抽象方法聲明使用,是必須被派生類覆寫的方法,抽象類就是用來被繼承的;能夠當作是沒有實現體的虛方法;若是類中包含抽象方法,那麼類就必須定義爲抽象類,不管是否還包含其餘通常方法;抽象類不能有實體的。spa
實例解答: 設計
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;
}
}