溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

C#中關(guān)于逆變和協(xié)變的示例分析

發(fā)布時(shí)間:2021-03-06 12:43:46 來源:億速云 閱讀:163 作者:小新 欄目:編程語言

這篇文章主要介紹C#中關(guān)于逆變和協(xié)變的示例分析,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!

在C#從誕生到發(fā)展壯大的過程中,新知識(shí)點(diǎn)不斷引入。逆變與協(xié)變并不是C#獨(dú)創(chuàng)的,屬于后續(xù)引入。在Java中同樣存在逆變與協(xié)變,后續(xù)我還會(huì)寫一篇Java逆變協(xié)變的文章,有興趣的朋友可以關(guān)注一下。

逆變與協(xié)變,聽起來很抽象、高深,其實(shí)很簡(jiǎn)單??聪旅娴拇a:

class Person
 {

 }
 class Student : Person
 {

 }
 class Teacher: Person
 {

 }
 
 class Program
 {
  static void Main(string[] args)
  {
   List<Person> plist = new List<Person>();
   plist = new List<Student>();
   plist = new List<Teacher>();
}
}

在上面的代碼中,plist = new List<Student>()、plist = new List<Teacher>()兩句產(chǎn)生編譯錯(cuò)誤。雖然Person是Student/Teacher的父類,但List<Person>類型卻不是List<Student/Teacher>類型的父類,所以上面的賦值語句報(bào)類型轉(zhuǎn)換失敗錯(cuò)誤。

如上這樣的賦值操作,在C# 4.0之前是不允許的,至于為什么不允許,類型安全是首要因素??聪旅娴氖纠a:

List<Person> plist = new List<Student>();
plist.Add(new Person());
plist.Add(new Student());
plist.Add(new Teacher());

如下示例,假設(shè) List<Person> plist = new List<Student>() 允許賦值,那plist雖然類型為L(zhǎng)ist<Person>集合,但實(shí)際指向確是List<Student>集合。plist.Add(new Person()),添加操作實(shí)際調(diào)用的是List<Student>.Add()。Person類型無法安全轉(zhuǎn)換為Student,所以這樣的集合定義沒有意義,所以上面的假設(shè)不成立。

但情況在C# 4.0之后發(fā)生了變化,并不是"不可能發(fā)生的事情發(fā)生了",而是應(yīng)用的靈活性做出了新的調(diào)整。同樣的在C# 4.0中上面的程序仍是不被允許的,但卻出現(xiàn)了例外。從C# 4.0開始,在泛型委托、泛型接口中,允許特殊情況的發(fā)生(實(shí)質(zhì)上并未發(fā)生特殊變化,后面說明)。如下示例:

delegate void Work<T>(T item);

class Person
{
  public string Name { get; set; }
}
class Student : Person
{
  public string Like { get; set; }
}
class Teacher : Person
{
  public string Teach { get; set; }
}

class Program
{
  static void Main(string[] args)
  {
   Work<Person> worker = (p) => { Console.WriteLine(p.Name); }; ;
   Work<Student> student_worker = (s) => { Console.WriteLine(s.Like); };
   student_worker = worker; //此處編譯錯(cuò)誤
  }
}

根據(jù)前面的理論支持,student_worker = worker;的錯(cuò)誤很容易理解。但此處我們程序的目的是讓 woker  充當(dāng) Work<Student> 的功能,以后調(diào)用 student_worker(s)實(shí)際調(diào)用的是woker(s)。為了滿足我們的需求,需要程序做2方面的處理:

1、因在調(diào)用student_worker(s)時(shí),實(shí)質(zhì)執(zhí)行的是woker(s),所以需要s變量的類型能成功轉(zhuǎn)換為woker需要的參數(shù)類型。

2、需要告訴編譯器,此處允許將 Work<Person> 類型的對(duì)象賦值給 Work<Student>類型的變量。

C#中關(guān)于逆變和協(xié)變的示例分析

條件1在調(diào)用時(shí)student_worker(),時(shí)編譯器會(huì)提示要求參數(shù)必須是Student類型對(duì)象,該對(duì)象可成功轉(zhuǎn)換為Person類型對(duì)象。

條件2則需要對(duì)Woke委托定義進(jìn)行調(diào)整,調(diào)整如下:

delegate void WorkIn<in T>(T item);

委托名字改為WorkIn是為卻別修改前后的委托,關(guān)鍵之處為<in T>。通過增加 in 關(guān)鍵字,標(biāo)注該泛型委托的類型參數(shù)T,僅作為委托方法的參數(shù)來使用。此時(shí)上面的程序便可成功編譯并執(zhí)行。

delegate void WorkIn<in T>(T item);
class Program
 {
  static void Main(string[] args)
  {
   WorkIn<Person> woker = (p) => { Console.WriteLine(p.Name); };
   WorkIn<Student> student_worker = woker;
   student_worker(new Student() { Name="tom", Like="C#" });

  }
 }

對(duì)于要求類型參數(shù)為子類型,允許賦值類型參數(shù)為父類型值的這種情況,稱為逆變。逆變?cè)贑#中需要用 in 標(biāo)注泛型的類型參數(shù)。逆變雖叫逆變,但只是形式上看似父類對(duì)象賦值給子類變量,實(shí)質(zhì)上是方法調(diào)用時(shí)參數(shù)的類型轉(zhuǎn)換。Student s = new Person(),這是不可能的,這不是逆變是錯(cuò)誤。

上面的代碼如你能轉(zhuǎn)換為下面的形式,那你就可以忘卻逆變,本質(zhì)比現(xiàn)象更重要

以上是“C#中關(guān)于逆變和協(xié)變的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對(duì)大家有幫助,更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI