C#逆變與協(xié)變?cè)斀?/h1>
來源:本站原創(chuàng)|時(shí)間:2020-01-10|欄目:C#教程|點(diǎn)擊: 次
該文章中使用了較多的 委托delegate和Lambda表達(dá)式,如果你并不熟悉這些,請(qǐng)查看我的文章《委托與匿名委托》、《匿名委托與Lambda表達(dá)式》以便幫你建立完整的知識(shí)體系。
在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>類型的變量。
條件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)象更重要😀:
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 = (s)=> { woker(s); };
student_worker(new Student() { Name="tom", Like="C#" });
}
}
協(xié)變
現(xiàn)在修改我們的程序需求,要求Work委托執(zhí)行后返回一個(gè)Person對(duì)象,如下:
delegate T Work<T>();
class Program
{
static void Main(string[] args)
{
Work<Person> worker = () => { return new Person(); };
Work<Student> student_worker = () => { return new Student(); };
worker = student_worker;
}
}
同上 worker = student_worker 無法通過編譯,此時(shí)我們的目的為:用 Work<Student> student_woker 的功能替代 Work<Person> 的功能,因?yàn)?student_woker 執(zhí)行后返回一個(gè)Student對(duì)象,這完全符合 Work<Person> 的要求。
如果要實(shí)現(xiàn)上面的目的,程序同樣需做2方面的處理:
1、因在調(diào)用 worker()時(shí),實(shí)質(zhì)執(zhí)行的是 student_worker(),所以需要 student_worker() 執(zhí)行結(jié)果能功轉(zhuǎn)換為woker 執(zhí)行后返回的類型。
2、需要告訴編譯器,此處允許將 Work<Student>類型的對(duì)象賦值給 Work<Person> 類型的變量。
此時(shí)條件1,上述代碼已經(jīng)滿足,對(duì)于條件2,需要泛型委托Work做如下調(diào)整:
delegate T WorkOut<out T>();
委托名字改為WorkOut也為卻別修改前后的委托,關(guān)鍵之處為<out T>。通過增加 out 關(guān)鍵字,標(biāo)注該泛型委托的類型參數(shù)T,僅作為委托方法的返回值類型來使用。此時(shí)上面的程序便可成功編譯并執(zhí)行。
delegate T WorkOut<out T>();
class Program
{
static void Main(string[] args)
{
WorkOut<Person> worker = () => { return new Person(); };
WorkOut<Student> student_worker = () => { return new Student(); };
worker = student_worker;
Person p = worker();
}
}
對(duì)于要求泛型類型參數(shù)為父類型,允許賦值類型參數(shù)為子類型值的這種情況,稱為協(xié)變。協(xié)變?cè)贑#中需要用 out 標(biāo)注泛型的類型參數(shù)。
注意:逆變、協(xié)變類型說明的區(qū)別。根據(jù)引出的定義逆變的形式只可能發(fā)生在泛型上(泛型接口、泛型委托),而協(xié)變的代碼形式就比較多,但并不一定是協(xié)變。所以在協(xié)變中用紅色注明,必須是關(guān)于泛型參數(shù)的情況才是協(xié)變。下面這類情況不屬于協(xié)變(至少我不認(rèn)為它們是協(xié)變):
Person p = new Student();
上面的示例代碼如你能轉(zhuǎn)換為下面的形式,那你也可以忘卻協(xié)變😀:
delegate T WorkOut<out T>();
class Program
{
static void Main(string[] args)
{
WorkOut<Student> student_worker = () => { return new Student(); };
WorkOut<Person> worker = () => { return student_worker (); };
Person p = worker();
}
}
通過上面的內(nèi)容可以發(fā)現(xiàn),逆變、協(xié)變其實(shí)是方法參數(shù)、返回值類型的轉(zhuǎn)換與對(duì)委托方法的包裝而已。抓住其核心,再看各種形式的代碼就簡(jiǎn)單了。
在C# 4.0 中 你可以查看 Action,F(xiàn)unc的定義,以便更深入理解逆變、協(xié)變。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
該文章中使用了較多的 委托delegate和Lambda表達(dá)式,如果你并不熟悉這些,請(qǐng)查看我的文章《委托與匿名委托》、《匿名委托與Lambda表達(dá)式》以便幫你建立完整的知識(shí)體系。
在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>類型的變量。
條件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)象更重要😀:
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 = (s)=> { woker(s); }; student_worker(new Student() { Name="tom", Like="C#" }); } }
協(xié)變
現(xiàn)在修改我們的程序需求,要求Work委托執(zhí)行后返回一個(gè)Person對(duì)象,如下:
delegate T Work<T>(); class Program { static void Main(string[] args) { Work<Person> worker = () => { return new Person(); }; Work<Student> student_worker = () => { return new Student(); }; worker = student_worker; } }
同上 worker = student_worker 無法通過編譯,此時(shí)我們的目的為:用 Work<Student> student_woker 的功能替代 Work<Person> 的功能,因?yàn)?student_woker 執(zhí)行后返回一個(gè)Student對(duì)象,這完全符合 Work<Person> 的要求。
如果要實(shí)現(xiàn)上面的目的,程序同樣需做2方面的處理:
1、因在調(diào)用 worker()時(shí),實(shí)質(zhì)執(zhí)行的是 student_worker(),所以需要 student_worker() 執(zhí)行結(jié)果能功轉(zhuǎn)換為woker 執(zhí)行后返回的類型。
2、需要告訴編譯器,此處允許將 Work<Student>類型的對(duì)象賦值給 Work<Person> 類型的變量。
此時(shí)條件1,上述代碼已經(jīng)滿足,對(duì)于條件2,需要泛型委托Work做如下調(diào)整:
delegate T WorkOut<out T>();
委托名字改為WorkOut也為卻別修改前后的委托,關(guān)鍵之處為<out T>。通過增加 out 關(guān)鍵字,標(biāo)注該泛型委托的類型參數(shù)T,僅作為委托方法的返回值類型來使用。此時(shí)上面的程序便可成功編譯并執(zhí)行。
delegate T WorkOut<out T>(); class Program { static void Main(string[] args) { WorkOut<Person> worker = () => { return new Person(); }; WorkOut<Student> student_worker = () => { return new Student(); }; worker = student_worker; Person p = worker(); } }
對(duì)于要求泛型類型參數(shù)為父類型,允許賦值類型參數(shù)為子類型值的這種情況,稱為協(xié)變。協(xié)變?cè)贑#中需要用 out 標(biāo)注泛型的類型參數(shù)。
注意:逆變、協(xié)變類型說明的區(qū)別。根據(jù)引出的定義逆變的形式只可能發(fā)生在泛型上(泛型接口、泛型委托),而協(xié)變的代碼形式就比較多,但并不一定是協(xié)變。所以在協(xié)變中用紅色注明,必須是關(guān)于泛型參數(shù)的情況才是協(xié)變。下面這類情況不屬于協(xié)變(至少我不認(rèn)為它們是協(xié)變):
Person p = new Student();
上面的示例代碼如你能轉(zhuǎn)換為下面的形式,那你也可以忘卻協(xié)變😀:
delegate T WorkOut<out T>(); class Program { static void Main(string[] args) { WorkOut<Student> student_worker = () => { return new Student(); }; WorkOut<Person> worker = () => { return student_worker (); }; Person p = worker(); } }
通過上面的內(nèi)容可以發(fā)現(xiàn),逆變、協(xié)變其實(shí)是方法參數(shù)、返回值類型的轉(zhuǎn)換與對(duì)委托方法的包裝而已。抓住其核心,再看各種形式的代碼就簡(jiǎn)單了。
在C# 4.0 中 你可以查看 Action,F(xiàn)unc的定義,以便更深入理解逆變、協(xié)變。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:C#實(shí)現(xiàn)簡(jiǎn)單的計(jì)算器功能完整實(shí)例
欄 目:C#教程
本文標(biāo)題:C#逆變與協(xié)變?cè)斀?/a>
本文地址:http://mengdiqiu.com.cn/a1/C_jiaocheng/5526.html
您可能感興趣的文章
- 01-10C#通過重寫Panel改變邊框顏色與寬度的方法
- 01-10C#實(shí)現(xiàn)實(shí)體類與字符串互相轉(zhuǎn)換的方法
- 01-10C#實(shí)現(xiàn)子窗體與父窗體通信方法實(shí)例總結(jié)
- 01-10時(shí)間戳與時(shí)間相互轉(zhuǎn)換(php .net精確到毫秒)
- 01-10基于C#實(shí)現(xiàn)簡(jiǎn)單離線注冊(cè)碼生成與驗(yàn)證
- 01-10C#開發(fā)中的垃圾回收機(jī)制簡(jiǎn)析
- 01-10C#編程實(shí)現(xiàn)對(duì)象與JSON串互相轉(zhuǎn)換實(shí)例分析
- 01-10C#多線程編程之使用ReaderWriterLock類實(shí)現(xiàn)多用戶讀與單用戶寫同步
- 01-10輕松學(xué)習(xí)C#的裝箱與拆箱
- 01-10C#定制Excel界面并實(shí)現(xiàn)與數(shù)據(jù)庫交互的方法


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 01-10C#通過反射獲取當(dāng)前工程中所有窗體并
- 01-10關(guān)于ASP網(wǎng)頁無法打開的解決方案
- 01-10WinForm限制窗體不能移到屏幕外的方法
- 01-10WinForm繪制圓角的方法
- 01-10C#實(shí)現(xiàn)txt定位指定行完整實(shí)例
- 01-10WinForm實(shí)現(xiàn)仿視頻播放器左下角滾動(dòng)新
- 01-10C#停止線程的方法
- 01-10C#實(shí)現(xiàn)清空回收站的方法
- 01-10C#通過重寫Panel改變邊框顏色與寬度的
- 01-10C#實(shí)現(xiàn)讀取注冊(cè)表監(jiān)控當(dāng)前操作系統(tǒng)已
隨機(jī)閱讀
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 04-02jquery與jsp,用jquery