日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷解決方案
一文搞定泛型,提高代碼復(fù)用率及程序的運(yùn)行性能

【稿件】泛型是程序設(shè)計(jì)語(yǔ)言的一種風(fēng)格,允許程序員在強(qiáng)類型程序設(shè)計(jì)語(yǔ)言中編寫代碼時(shí)使用一些以后才指定的類型,在實(shí)例化時(shí)作為參數(shù)指明這些類型。泛型在 .NET 中應(yīng)用尤其廣泛,泛型是在 .NET 2.0 CLR 中的增加的一項(xiàng)新功能,類似于 C++ 的模板但不如 C++ 的模板靈活,不過也有一些自己的特性。泛型為 .NET 引入了類型參數(shù)的概念,這樣便可以把指定類型的工作推遲到客戶端代碼聲明并實(shí)例化類或方法的時(shí)候執(zhí)行。下面我們就來講解一下泛型的知識(shí)。

成都創(chuàng)新互聯(lián)公司服務(wù)項(xiàng)目包括開魯網(wǎng)站建設(shè)、開魯網(wǎng)站制作、開魯網(wǎng)頁(yè)制作以及開魯網(wǎng)絡(luò)營(yíng)銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢(shì)、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,開魯網(wǎng)站推廣取得了明顯的社會(huì)效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到開魯省份的部分城市,未來相信會(huì)繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!

一、當(dāng) C# 沒有泛型

在 .NET 2.0 以前沒有泛型的時(shí)候,開發(fā)人員一直在使用 System.Collections.Stack 類,它是一個(gè)棧類型的集合對(duì)象。 Stack 通過 Push 和 Pop 方法向集合中添加和刪除數(shù)據(jù)。很多開發(fā)人員通過前面的描述都會(huì)認(rèn)為使用 Stack 很簡(jiǎn)單,但是其中存在一個(gè)重大的缺陷。 Stack 類所保存的是 object 類型,這樣就導(dǎo)致了 CLR 無(wú)法驗(yàn)證 push 進(jìn)集合中的對(duì)象是不是想要的類型。

此外當(dāng)我們使用 Pop 方法時(shí)需要將它的返回值轉(zhuǎn)換為我們需要的類型,因此這里就存在一個(gè)問題,如果 Pop 方法的返回值不是我們需要的類型那么就有很大可能引發(fā)異常。這里的返回值轉(zhuǎn)換使用的是強(qiáng)制類型轉(zhuǎn)換,由于使用了強(qiáng)制類型轉(zhuǎn)換將類型檢查放在了運(yùn)行時(shí)進(jìn)行,因此代碼就變得更加脆弱。

使用 Stack 類還存在一個(gè)性能問題,將值類型的實(shí)例傳遞給 Push 方法,運(yùn)行時(shí)將會(huì)對(duì)它進(jìn)行裝箱操作,頻繁的執(zhí)行值類型裝箱操作,系統(tǒng)會(huì)頻繁的分配內(nèi)存、復(fù)制值已經(jīng)進(jìn)行垃圾回收,這樣就導(dǎo)致了大量的性能開銷。通過前面的描述部分讀者應(yīng)該看出來了 Stack 類不是類型安全的類,因此在不使用泛型的情況下,我們?nèi)绻薷?Stack 類并保證它是類型安全的,并且要求它存儲(chǔ)指定的類型的話,我們必須這么做:

 
 
 
  1. public class StackDemo  
  2. {  
  3. public virtual User Pop();  
  4. public virtual void Push(User user);  
  5. //more code  

上面的代碼是不是很簡(jiǎn)單?如果你真的這么認(rèn)為那么你就是想多了,由于我們要求只能存儲(chǔ) User 類型的隊(duì)形,因此我們需要對(duì) Stack 的每個(gè)方法進(jìn)行重寫實(shí)現(xiàn),如果我們還需要一個(gè)存儲(chǔ) Student 類型的 Stack ,我們就需要再重寫一次 Stack 的每個(gè)方法。這就凸顯了一個(gè)問題,代碼中產(chǎn)生了大量的類似的代碼和重復(fù)的代碼。

另外在沒有泛型的情況下如果聲明允許包含 Null 值的變量的時(shí)候就比較麻煩了。一般情況下我們常用的有兩種方法。

方法一:對(duì)需要處理 null 值的每個(gè)類型都需要聲明可空數(shù)據(jù)類型,我們來看個(gè)簡(jiǎn)單的例子:

 
 
 
  1. struct NullInt  
  2. {  
  3. public int Value { get; private set;}  
  4. public bool HasValue {get; private set;}  

上述例子很簡(jiǎn)單,但是存在兩個(gè)問題,首先如果我們有很多可空類型的話,我們就需要編寫大量的類似代碼,其次如果可空值類型發(fā)生了改變那么我們就必須修改所有的可能類型聲明,可想而知工作量是非常巨大的,而且也很容易出現(xiàn)紕漏和錯(cuò)誤。

方法二:這個(gè)方法的出現(xiàn)就是為了解決我們?cè)诜椒ㄒ恢兴岬降膬蓚€(gè)問題。我們只需要聲明一個(gè)可能類型即可,類型中包含 object 類型的 Value 屬性,同樣我們先來看一下代碼:

 
 
 
  1. struct NullType  
  2. {  
  3. public object Value {get; private set;}  
  4. public bool HasValue {get; private set;}  

這個(gè)方法雖然充分解決了方法一出現(xiàn)的問題,但是它并不完美。因?yàn)檫\(yùn)行時(shí)在設(shè)置 Value 屬性的時(shí)候總是會(huì)對(duì)值類型進(jìn)行裝箱,另外通過 NullType.Value 獲取值地時(shí)候需要進(jìn)行強(qiáng)制類型裝換,這個(gè)操作在運(yùn)行時(shí)可能會(huì)報(bào)錯(cuò)。

二、泛型概述

泛型類型是 C# 2.0 引入的,它的引入在一定程度上減輕了開發(fā)人員的壓力,同時(shí)也使得程序變得更加健壯和穩(wěn)定。泛型類的語(yǔ)法也很簡(jiǎn)單,用尖括號(hào)聲明泛型類型參數(shù)和提供泛型類型實(shí)參即可。我們來看一個(gè)定義泛型的例子:

 
 
 
  1. public class DataBaseOperating  
  2. {  
  3. private T[] ModelArray{get;}  
  4. public bool Insert(T t)  
  5. {  
  6. //more code  
  7. }  
  8. public T SelectOne()  
  9. {  
  10. //more code  
  11. }  
  12. public List SelectAll()  
  13. {  
  14. //more code  
  15. }  
  16. //more code  

前面這段代碼,我們定義了操作數(shù)據(jù)庫(kù)的泛型類,這個(gè)類可以被項(xiàng)目中所有需要操作數(shù)據(jù)庫(kù)的類使用,我們只需將類型實(shí)參傳遞進(jìn)來即可。例如我們需要向數(shù)據(jù)庫(kù)插入一條 User 數(shù)據(jù)。我們可以這么做:

 
 
 
  1. //more code  
  2. DataBaseOperating dbOp=new DataBaseOperating();  
  3. dbOp.Insert(user);  
  4. //more code 

我們看到在定義泛型類的時(shí)候,我定義類型參數(shù)用的是 T ,這么做是大部分 C# 開發(fā)人員的一個(gè)習(xí)慣,也可以說是一個(gè)大家都默認(rèn)的規(guī)范,我們?cè)陂_發(fā)時(shí)一般都會(huì)使用以大寫字母 T 作為前綴來表明它是一個(gè)類型參數(shù)。泛型的定義和使用就這么多,是不是很簡(jiǎn)單呢?下面我們就來講解一下泛型的各個(gè)方面。在學(xué)習(xí)泛型類之前我們要先來了解一下它的優(yōu)點(diǎn),來看看為什么微軟在 C# 2.0 中引入了泛型類。

泛型促進(jìn)了類型安全,它確保了參數(shù)化類中只有成員明確希望的數(shù)據(jù)類型才可以使用;

類型檢查會(huì)在編譯時(shí)發(fā)生進(jìn)而減少了在運(yùn)行時(shí)出現(xiàn)強(qiáng)制類型轉(zhuǎn)換無(wú)效的錯(cuò)誤;

泛型類成員使用的是值類型,因此就不會(huì)出現(xiàn) object 裝箱轉(zhuǎn)換操作。并且代碼既保持具體類的優(yōu)勢(shì)又避免了具體類的開銷,這樣代碼的性能得以提高,內(nèi)存消耗也變得很少。

構(gòu)造函數(shù)

我們?cè)陂_發(fā)中經(jīng)常用到構(gòu)造函數(shù),在泛型類和泛型結(jié)構(gòu)中同樣也適用構(gòu)造函數(shù)。泛型類/結(jié)構(gòu)的構(gòu)造函數(shù)和普通類/結(jié)構(gòu)的構(gòu)造函數(shù)是一模一樣的,不需要類型參數(shù)只需要按照普通類/結(jié)構(gòu)的構(gòu)造函數(shù)定義方法定義即可。

 
 
 
  1. public class Demo  
  2. {  
  3. public T key {get;set;}  
  4. public Demo(T t)  
  5. {  
  6. key=t;  
  7. }  
  8. }  
  9. public struct Demo  
  10. {  
  11. public T value {get;set;}  
  12. public Demo(T t)  
  13. {  
  14. value=t;  
  15. }  

Tip:構(gòu)造函數(shù)包含類型參數(shù)也可以

結(jié)構(gòu)與接口

在 C# 中不僅僅存在泛型類,還存在泛型接口和泛型結(jié)構(gòu)。泛型接口和泛型結(jié)構(gòu)的語(yǔ)法和泛型類相同。這里主要講解一下在類中多次實(shí)現(xiàn)同一個(gè)泛型接口。我們先來看一下代碼:

 
 
 
  1. public interface IDemo  
  2. {  
  3. ICollection items {get;set;}  
  4. }  
  5. public class Demo:IDemo,IDemo  
  6. {  
  7. ICollection IDemo.items {get;set;}  
  8. ICollection IDemo.items {get;set;}  

在上述代碼中,我們?cè)陬愔酗@示實(shí)現(xiàn)了兩個(gè)不同類型實(shí)參的同一個(gè)泛型接口,一般來說在類中多次實(shí)現(xiàn)泛型接口并非是一個(gè)最優(yōu)的選擇,因?yàn)樗鼤?huì)造成代碼的混淆以及在使用的過程中造成誤會(huì)。因此除非特殊情況,絕大多數(shù)情況下,我們不應(yīng)該在一個(gè)類中多次實(shí)現(xiàn)同一個(gè)接口。

默認(rèn)值

當(dāng)我們需要在泛型類的構(gòu)造函數(shù)中部分屬性進(jìn)行初始化,而其他屬性不進(jìn)行初始化,但是我們?cè)陂_發(fā)中無(wú)法確定傳入泛型類中的類型參數(shù)是什么,因此我們也無(wú)法通過具體的值設(shè)置默認(rèn)值。這種情況在 C# 中可以說是非常好解決,我們可以調(diào)用 default 操作符來給傳入的任意類型參數(shù)提供默認(rèn)值。例如下面這段代碼,我們只初始化 Key ,Value 的初始化則利用 default 操作符。

 
 
 
  1. public class Demo  
  2. {  
  3. T tKey {get;set;}  
  4. T tValue {get;set;}  
  5. public Demo(T key)  
  6. {  
  7. tKey=key;  
  8. tValue=default(T);  
  9. }  

Tip:default 中的參數(shù)并非是必須傳入的,在 C#7 中如果可以推斷出數(shù)據(jù)類型的話是不需要指定參數(shù)的。比如 Demo demo = default(T) 就可以寫成 Demo demo=default。

多類型參數(shù)

前面我們所講的都是單個(gè)類型參數(shù)的泛型類,但是泛型類型不僅僅只能具有一個(gè)參數(shù),它可以具有無(wú)限多的參數(shù),例如我們定義一個(gè)泛型類,它的構(gòu)造函數(shù)接受兩個(gè)不同類型的參數(shù),代碼可以這么實(shí)現(xiàn)。

 
 
 
  1. public class Demo() 
  2. public TKey key {get;set;} 
  3. public TValue value {get;set;} 
  4. public Demo(TKey tKey,TValue tValue) 
  5. key=tKey; 
  6. value=tValue; 

我們?cè)谑褂?Demo 時(shí),只需要聲明和實(shí)例化語(yǔ)句尖括號(hào)中指定的多個(gè)類型參數(shù)即可。在調(diào)用時(shí)要提供和方法參數(shù)匹配的類型。

 
 
 
  1. Demo demo=new Demo(1,"小明");  
  2. Console.Write($"編號(hào) {demo.key} 是 {demo.value}") 

Tip:在 C# 中在同一個(gè)命名空間中可以存在多個(gè)同名但類型參數(shù)數(shù)量不同的類。在部分文章或圖書中會(huì)將類型參數(shù)數(shù)量稱為 元數(shù) 。

嵌套泛型類型

嵌套泛型類型在開發(fā)中用的比較少,但是還是有必要在這里說一下,因?yàn)橛胁糠珠_發(fā)人員對(duì)于這一塊不甚了解。嵌套泛型類型的外層也是一個(gè)泛型類型,外層的這個(gè)泛型類型通常被稱為包容泛型類型,嵌套泛型類型會(huì)自動(dòng)獲得包容泛型類型的類型參數(shù),這段話有些繞口,我詳細(xì)講解一下。

例如 A 是包容泛型類型,它有一個(gè)類型參數(shù) T,B 是嵌套泛型類型,它位于 A 中,那么 B 也可以使用 A 的類型參數(shù) T ,如果 B 中也包含一個(gè)類型參數(shù) T ,那么 B 會(huì)隱藏 A 的類型參數(shù) T 。這里需要提醒的是如果嵌套泛型類型的類型參數(shù)和包容泛型類型的類型參數(shù)相同,那么開發(fā)工具將會(huì)出現(xiàn)編譯警告,這個(gè)警告是在告知開發(fā)人員使用了相同的類型參數(shù),因此這里就引出一條編碼規(guī)則:避免在嵌套泛型類型中使用同名參數(shù)隱藏外層類型的類型參數(shù)。

泛型方法

前面我們所說的都是泛型類,在 C# 中除了有泛型類還有泛型方法,泛型方法的語(yǔ)法和泛型類的語(yǔ)法類似,并且泛型方法不僅可以出現(xiàn)在泛型類中也可以出現(xiàn)在普通類中。泛型方法和泛型類相比有一個(gè)很特別的地方,就是泛型方法可以自己推斷類型。編譯器可以根據(jù)傳給方法的實(shí)參來推斷泛型參數(shù)類型。因此如果想讓方法類型推斷成功,那么實(shí)參類型必須與泛型方法的形參相匹配。

三、泛型約束

在開發(fā)中大部分情況,我們不允許任何不符合我們要求的類型參數(shù)出現(xiàn)在我們的代碼中并引起錯(cuò)誤。要杜絕這個(gè)問題就需要用到泛型約束。聲明泛型約束需要使用 where 關(guān)鍵字,后面跟一對(duì) 參數(shù):要求 。這里面的參數(shù)必須是泛型類型中聲明的一個(gè)參數(shù),要求描述的是類型參數(shù)所能轉(zhuǎn)換成的類或接口等條件。泛型約束分為:接口約束、類類型約束、class 和 struct 約束、多約束以及構(gòu)造函數(shù)約束。下面我們就來一一講解一下。

接口約束

為規(guī)定某個(gè)數(shù)據(jù)類型必須是向某個(gè)接口,需要聲明一個(gè) 接口類型約束 ,利用這種約束可以避免需要通過轉(zhuǎn)型才能調(diào)用一個(gè)顯示接口成員的實(shí)現(xiàn)。下面我們通過一個(gè)代碼段來講解一下接口約束。

 
 
 
  1. public class Demo where T:System.IComparable  
  2. {  
  3. //more code  

在上面這段代碼中,我們添加了 System.IComparable 約束,也就是說所提供的類型參數(shù)都必須實(shí)現(xiàn) System.IComparable 接口。那么當(dāng)我們向 Demo 傳遞 StringBuilder 作為類型參數(shù)來創(chuàng)建 Demo 變量時(shí)編譯器會(huì)報(bào)告一個(gè)錯(cuò)誤,這是因?yàn)?StringBuilder 沒有實(shí)現(xiàn) IComparable 接口。

類類型約束

當(dāng)我們需要將類型實(shí)參轉(zhuǎn)換為特定的類類型時(shí)就需要用到 類類型約束。類類型約束的語(yǔ)法和接口約束語(yǔ)法相同。這里有一點(diǎn)需要注意,如果同時(shí)指定了多種約束,那么類類型約束必須位于第一位(第一個(gè)出現(xiàn)),并且泛型約束中是不允許使用多個(gè)類類型約束的,這是因?yàn)槲覀兊拇a不可能從多個(gè)不想管的類中派生出來,同樣類類型約束也不能指定密封類或者不是類的類型。

class、struct 約束

class 和 struct 約束是一個(gè)很容易出錯(cuò)并且也很容易讓新手程序員造成困惑的地方。首先很多新手程序員看到 class 約束會(huì)認(rèn)為是將類型實(shí)參限制為類類型,其實(shí)不是這樣的。class 約束是類型實(shí)參為引用類型,因此這里使用接口、類、委托以技術(shù)組類型都符合這個(gè)條件。struct 約束和 class 約束正好相反,它是將類型實(shí)參限制為值類型,并且值類型還不能是可空值類型。因?yàn)榭煽罩殿愋褪亲鳛榉盒?NUllable 來實(shí)現(xiàn)的,并且 NUllable 中的 T 使用的是 struct 約束。如果可以使用可控制類型那么 NUllable 就有很大的可能被用成 NUllable

Tip:因?yàn)?class 約束要求引用類型而 struct 約束要求值類型,因此這兩種約束是不能同時(shí)出現(xiàn)的。

多約束

我們可以為任意類型的參數(shù)指定任意數(shù)量的接口約束,所有的接口約束需要用逗號(hào)分割。如果存在多個(gè)不同類型的約束,針對(duì)每種約束都需要寫一個(gè) where 關(guān)鍵字,不同種類約束之間不需要用任何符號(hào)分割。我們來看一下多約束的代碼段:

 
 
 
  1. public class Demo  
  2. where TKey:IA,IB  
  3. where TValue: ClassA  
  4. {  
  5. //more code  

構(gòu)造函數(shù)約束

有時(shí)我們需要在泛型類中創(chuàng)建類型實(shí)參的實(shí)例,這時(shí)我們可以規(guī)定傳入泛型類的類型實(shí)參必須具有構(gòu)造函數(shù),如果要實(shí)現(xiàn)這一點(diǎn)我們可以使用new() 來作為限制,這個(gè)約束叫做 構(gòu)造函數(shù)約束 。這里需要注意的是構(gòu)造函數(shù)約束必須位于所有約束的后面,并且它只能對(duì)默認(rèn)構(gòu)造函數(shù)進(jìn)行約束,而不能對(duì)有參構(gòu)造函數(shù)進(jìn)行約束。

Tip 1:關(guān)于約束繼承這個(gè)問題,想必好多開發(fā)人員都是一頭霧水。在這里我通過簡(jiǎn)單的幾句來說一下約束繼承。首先無(wú)論是泛型類型參數(shù)還是它們的約束都不會(huì)被 派生類 繼承,這是因?yàn)榉盒皖愋蛥?shù)和約束不是類的成員。雖然不能被派生類繼承,但是可以被從其派生的泛型類所繼承。由于派生的泛型類類型參數(shù)是泛型基類的類型實(shí)參,所以類型參數(shù)必須具有等同于或者強(qiáng)于泛型基類的約束條件。

Tip 2:泛型方法同樣也可以使用約束,約束條件和泛型類類似。

六、總結(jié)

這篇文章我主要講解了泛型的一些知識(shí),不能說很全面,但已經(jīng)覆蓋了百分之九十的內(nèi)容。泛型在開發(fā)中可以說是經(jīng)常用到,良好的使用泛型可以提高代碼復(fù)用率以及程序的運(yùn)行性能。

作者介紹

朱鋼,筆名喵叔,國(guó)內(nèi)知名技術(shù)社區(qū)博客認(rèn)證專家,2019年知名技術(shù)社區(qū)博客之星20強(qiáng),.NET高級(jí)開發(fā)工程師,7年一線開發(fā)經(jīng)驗(yàn),參與過電子政務(wù)系統(tǒng)和AI客服系統(tǒng)的開發(fā),以及互聯(lián)網(wǎng)招聘網(wǎng)站的架構(gòu)設(shè)計(jì),目前就職于一家初創(chuàng)公司,從事企業(yè)級(jí)安全監(jiān)控系統(tǒng)的開發(fā)。

【原創(chuàng)稿件,合作站點(diǎn)轉(zhuǎn)載請(qǐng)注明原文作者和出處為.com】


網(wǎng)站欄目:一文搞定泛型,提高代碼復(fù)用率及程序的運(yùn)行性能
文章鏈接:http://m.5511xx.com/article/cdeojch.html