C#子類對(duì)基類方法的繼承、重寫(xiě)與隱藏詳解
前言
提起子類、基類和方法繼承這些概念,肯定大家都非常熟悉。畢竟,作為一門(mén)支持OOP的語(yǔ)言,掌握子類、基類是學(xué)習(xí)C#的基礎(chǔ)。不過(guò),這些概念雖然簡(jiǎn)單,但是也有一些初學(xué)者可能會(huì)遇到的坑,我們一起看看吧。
子類繼承基類非私有方法
首先我們看最簡(jiǎn)單的一種,子類繼承自基類,但子類對(duì)繼承的方法沒(méi)有任何改動(dòng)
class Person
{
public void Greeting()
{
Console.WriteLine("Hello, I am Person");
}
}
class Employee : Person
{
}
class Program
{
static void Main(string[] args)
{
Person p = new Employee();
p.Greeting();
}
}
在這個(gè)例子中,作為子類的Employee自動(dòng)繼承了基類的Greeting方法,當(dāng)在子類實(shí)例調(diào)用這個(gè)方法的時(shí)候,實(shí)際上調(diào)用的是基類的方法。這個(gè)例子非常簡(jiǎn)單,毋庸多言。
子類覆蓋基類方法
接著是最常見(jiàn)的情況,子類覆蓋基類的方法,典型的例子如下
class Person
{
public virtual void Greeting()
{
Console.WriteLine("Hello, I am Person");
}
}
class Employee : Person
{
public override void Greeting()
{
Console.WriteLine("Hello, I am Employee");
}
}
class Program
{
static void Main(string[] args)
{
Employee e = new Employee();
Person p = e;
p.Greeting();
e.Greeting();
}
}
同樣,這段代碼也很簡(jiǎn)單,基類方法通過(guò)關(guān)鍵字virtual表明方法可以被覆蓋,子類通過(guò)關(guān)鍵字override實(shí)現(xiàn)對(duì)基類方法的覆蓋,最后看調(diào)用部分,無(wú)論變量類型是子類還是基類,只要對(duì)象實(shí)際類型是子類,調(diào)用的方法都是子類覆蓋的方法,這也是多態(tài)的實(shí)現(xiàn)基礎(chǔ)。
子類隱藏基類方法
上面兩個(gè)例子都非常簡(jiǎn)單,邏輯也很清楚,有點(diǎn)繞的要算子類隱藏基類方法的情況。
子類隱藏基類的非虛方法
基類被子類繼承的方法可能是虛方法,也可能是非虛方法,先看非虛方法被子類隱藏的情況,隱藏基類方法使用的關(guān)鍵字是new
class Person
{
public void Greeting()
{
Console.WriteLine("Hello, I am Person");
}
}
class Employee : Person
{
public new void Greeting()
{
Console.WriteLine("Hello, I am Employee");
}
}
class Program
{
static void Main(string[] args)
{
Employee e = new Employee();
Person p = e;
p.Greeting();
e.Greeting();
}
}

這里的結(jié)果可能就出乎某些初學(xué)者的意料了,為什么明明是子類Employee的實(shí)例,卻在不同的引用變量類型下呈現(xiàn)出了不一樣的效果?為什么會(huì)調(diào)用到了基類里面的方法?
其實(shí)這跟C#的函數(shù)調(diào)用機(jī)制有關(guān),一般來(lái)說(shuō),C#編譯成MSIL之后,有兩種函數(shù)調(diào)用方式。
- Call 以非虛的方式調(diào)用方法,一般用于靜態(tài)函數(shù)調(diào)用,因?yàn)殪o態(tài)函數(shù)不可能是虛的,但也可以以非虛的方式調(diào)用一個(gè)虛方法
- Callvirt 以虛方式調(diào)用,一般用于非靜態(tài)方法和虛方法的調(diào)用。如果調(diào)用的方法非虛,則引用變量類型決定了最終調(diào)用的方法;反之,如果調(diào)用的方法為虛,則實(shí)例變量類型決定最終調(diào)用的方法——因?yàn)榭赡艹霈F(xiàn)方法重寫(xiě),即,多態(tài)
用ILDASM打開(kāi)我們的程序集看看,

證明了這里確實(shí)是用的Callvirt,而這個(gè)方法是非虛的方法,所以在兩次調(diào)用中,引用變量類型Person和Employee就能夠決定所調(diào)用的方法。兩個(gè)類分別實(shí)現(xiàn)了自己的Greeting方法,沒(méi)有出現(xiàn)子類覆蓋基類方法的情況。這就解釋了為什么兩次調(diào)用結(jié)果不同。最后讓我們來(lái)看看最復(fù)雜的一種情況
子類隱藏基類的虛方法
考慮下面的代碼
class Person
{
public virtual void Greeting()
{
Console.WriteLine("Hello, I am Person");
}
}
class Employee : Person
{
public new virtual void Greeting()
{
Console.WriteLine("Hello, I am Employee");
}
}
class Manager : Employee
{
public override void Greeting()
{
Console.WriteLine("Hello, I am Manager");
}
}
class Program
{
static void Main(string[] args)
{
Manager m = new Manager();
Person p = m;
Employee e = m;
p.Greeting();
e.Greeting();
m.Greeting();
}
}
猜一下輸出應(yīng)該是什么?這也是老胡曾經(jīng)遇到過(guò)的一道筆試題,表面看著簡(jiǎn)單,但是不注意也會(huì)掉坑里
1,2,3,答案揭曉

是不是有點(diǎn)出乎意料呢,讓我們來(lái)分析一下

首先,三次調(diào)用均是callvirt,而且方法Greeting是虛方法,我們需要考慮對(duì)象實(shí)例以決定要調(diào)用的方法。
- 在第一次調(diào)用中,引用變量類型是Person,雖然對(duì)象實(shí)例類型Manger重寫(xiě)了Greeting方法,但是它重寫(xiě)的是繼承自Manger基類Emplyee的Greeting方法,Person中Greeting方法在子類Manger中僅僅是被隱藏而沒(méi)有被重寫(xiě),所以這里調(diào)用的是Person中的Greeting
- 而第二次調(diào)用中,引用變量類型是Employee,Employee的Greeting方法被Manager重寫(xiě),所以這次調(diào)用到的是Manager中的Greeting
- 最后一次調(diào)用毋庸多言,簡(jiǎn)單的重寫(xiě)案例而已
怎么樣,是不是有小伙伴猜錯(cuò)結(jié)果了?
總結(jié)
在子類對(duì)基類有方法繼承、重寫(xiě)和隱藏的情況下,有時(shí)候判斷具體哪個(gè)方法被調(diào)用會(huì)有難度,但請(qǐng)記住以下要點(diǎn):
如果被調(diào)用方法非虛,那么只用關(guān)注引用變量類型就好,引用變量類型能決定調(diào)用方法在哪里如果調(diào)用方法為虛,我們需要站在引用變量類型的角度,審視該方法是否被對(duì)象類型所重寫(xiě);若是,則調(diào)用對(duì)象類型的重寫(xiě)方法;反之,則再次讓引用變量類型決定調(diào)用方法。
這樣,當(dāng)我們?cè)儆龅阶宇愲[藏基類虛方法的情況,應(yīng)用以上要點(diǎn)就可以撥云見(jiàn)日。
到此這篇關(guān)于C#子類對(duì)基類方法的繼承、重寫(xiě)與隱藏的文章就介紹到這了,更多相關(guān)C#子類對(duì)基類方法繼承、重寫(xiě)與隱藏內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
使用C#編寫(xiě)兩個(gè)漂亮?xí)r鐘的示例代碼
這篇文章主要為大家分享了兩個(gè)使用C#編寫(xiě)的兩個(gè)漂亮?xí)r鐘的示例代碼,文中的示例代碼講解詳細(xì),具有一定的參考價(jià)值,感興趣的可以了解一下2023-07-07
舊項(xiàng)目升級(jí)新版Unity2021導(dǎo)致Visual?Studio無(wú)法使用的問(wèn)題
在項(xiàng)目開(kāi)發(fā)過(guò)程中,不可避免的會(huì)升級(jí)開(kāi)發(fā)工具。這次我在舊項(xiàng)目版本升級(jí)到新版Unity2021.2.x時(shí),出現(xiàn)Visual?Studio無(wú)法定位等問(wèn)題,這里我給大家分享下解決方法,舊項(xiàng)目升級(jí)新版Unity2021導(dǎo)致Visual?Studio無(wú)法使用的問(wèn)題,需要的朋友可以參考下2021-12-12
WCF實(shí)現(xiàn)的計(jì)算器功能實(shí)例
這篇文章主要介紹了WCF實(shí)現(xiàn)的計(jì)算器功能,結(jié)合具體實(shí)例形式較為詳細(xì)的分析了WCF實(shí)現(xiàn)計(jì)算器功能的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-06-06

