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

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

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
C#類(lèi)型參數(shù)約束分析及應(yīng)用淺析

C# 類(lèi)型參數(shù)約束使用的原因:如果要檢查泛型列表中的某個(gè)項(xiàng)以確定它是否有效,或者將它與其他某個(gè)項(xiàng)進(jìn)行比較,則編譯器必須在一定程度上保證它需要調(diào)用的運(yùn)算符或方法將受到客戶(hù)端代碼可能指定的任何類(lèi)型參數(shù)的支持。這種保證是通過(guò)對(duì)泛型類(lèi)定義應(yīng)用一個(gè)或多個(gè)約束獲得的。例如,基類(lèi)約束告訴編譯器:僅此類(lèi)型的對(duì)象或從此類(lèi)型派生的對(duì)象才可用作類(lèi)型參數(shù)。一旦編譯器有了這個(gè)保證,它就能夠允許在泛型類(lèi)中調(diào)用該類(lèi)型的方法。約束是使用上下文關(guān)鍵字 where 應(yīng)用的。下面的代碼示例演示可通過(guò)應(yīng)用基類(lèi)約束添加到 GenericList 類(lèi)的功能。

十載的隆林網(wǎng)站建設(shè)經(jīng)驗(yàn),針對(duì)設(shè)計(jì)、前端、開(kāi)發(fā)、售后、文案、推廣等六對(duì)一服務(wù),響應(yīng)快,48小時(shí)及時(shí)工作處理。成都全網(wǎng)營(yíng)銷(xiāo)推廣的優(yōu)勢(shì)是能夠根據(jù)用戶(hù)設(shè)備顯示端的尺寸不同,自動(dòng)調(diào)整隆林建站的顯示方式,使網(wǎng)站能夠適用不同顯示終端,在瀏覽器中調(diào)整網(wǎng)站的寬度,無(wú)論在任何一種瀏覽器上瀏覽網(wǎng)站,都能展現(xiàn)優(yōu)雅布局與設(shè)計(jì),從而大程度地提升瀏覽體驗(yàn)。創(chuàng)新互聯(lián)從事“隆林網(wǎng)站設(shè)計(jì)”,“隆林網(wǎng)站推廣”以來(lái),每個(gè)客戶(hù)項(xiàng)目都認(rèn)真落實(shí)執(zhí)行。

在定義泛型類(lèi)時(shí),可以對(duì)客戶(hù)端代碼能夠在實(shí)例化類(lèi)時(shí)用于類(lèi)型參數(shù)的類(lèi)型種類(lèi)施加限制。如果客戶(hù)端代碼嘗試使用某個(gè)約束所不允許的類(lèi)型來(lái)實(shí)例化類(lèi),則會(huì)產(chǎn)生編譯時(shí)錯(cuò)誤。這些限制稱(chēng)為約束。約束是使用 where 上下文關(guān)鍵字指定的。下面列出了六種類(lèi)型的約束:

◆T:結(jié)構(gòu)

類(lèi)型參數(shù)必須是值類(lèi)型??梢灾付ǔ?Nullable 以外的任何值類(lèi)型。

◆T:類(lèi)

類(lèi)型參數(shù)必須是引用類(lèi)型,包括任何類(lèi)、接口、委托或數(shù)組類(lèi)型。

◆T:new()

類(lèi)型參數(shù)必須具有無(wú)參數(shù)的公共構(gòu)造函數(shù)。當(dāng)與其他約束一起使用時(shí),new() 約束必須***指定。

◆T:<基類(lèi)名>

類(lèi)型參數(shù)必須是指定的基類(lèi)或派生自指定的基類(lèi)。

◆T:<接口名稱(chēng)>

類(lèi)型參數(shù)必須是指定的接口或?qū)崿F(xiàn)指定的接口。可以指定多個(gè)接口約束。約束接口也可以是泛型的。

◆T:U

為 T 提供的類(lèi)型參數(shù)必須是為 U 提供的參數(shù)或派生自為 U 提供的參數(shù)。這稱(chēng)為裸類(lèi)型約束。

C# 類(lèi)型參數(shù)約束代碼

 
 
 
  1. public class Employee  
  2. {  
  3.     private string name;  
  4.     private int id;  
  5.  
  6.     public Employee(string s, int i)  
  7.     {  
  8.         name = s;  
  9.         id = i;  
  10.     }  
  11.  
  12.     public string Name  
  13.     {  
  14.         get { return name; }  
  15.         set { name = value; }  
  16.     }  
  17.  
  18.     public int ID  
  19.     {  
  20.         get { return id; }  
  21.         set { id = value; }  
  22.     }  
  23. }  
  24.  
  25. public class GenericList  where T : Employee  
  26. {  
  27.     private class Node  
  28.     {  
  29.         private Node next;  
  30.         private T data;  
  31.  
  32.         public Node(T t)  
  33.         {  
  34.             next = null;  
  35.             data = t;  
  36.         }  
  37.  
  38.         public Node Next  
  39.         {  
  40.             get { return next; }  
  41.             set { next = value; }  
  42.         }  
  43.  
  44.         public T Data  
  45.         {  
  46.             get { return data; }  
  47.             set { data = value; }  
  48.         }  
  49.     }  
  50.  
  51.     private Node head;  
  52.  
  53.     public GenericList() //constructor  
  54.     {  
  55.         head = null;  
  56.     }  
  57.  
  58.     public void AddHead(T t)  
  59.     {  
  60.         Node n = new Node(t);  
  61.         n.Next = head;  
  62.         head = n;  
  63.     }  
  64.  
  65.     public IEnumerator  GetEnumerator()  
  66.     {  
  67.         Node current = head;  
  68.  
  69.         while (current != null)  
  70.         {  
  71.             yield return current.Data;  
  72.             current = current.Next;  
  73.         }  
  74.     }  
  75.  
  76.     public T FindFirstOccurrence(string s)  
  77.     {  
  78.         Node current = head;  
  79.         T t = null;  
  80.  
  81.         while (current != null)  
  82.         {  
  83.             //The constraint enables access to the Name property.  
  84.             if (current.Data.Name == s)  
  85.             {  
  86.                 t = current.Data;  
  87.                 break;  
  88.             }  
  89.             else 
  90.             {  
  91.                 current = current.Next;  
  92.             }  
  93.         }  
  94.         return t;  
  95.     }  

約束使得泛型類(lèi)能夠使用 Employee.Name 屬性,因?yàn)轭?lèi)型為 T 的所有項(xiàng)都保證是 Employee 對(duì)象或從 Employee 繼承的對(duì)象。

可以對(duì)同一類(lèi)型參數(shù)應(yīng)用多個(gè)約束,并且約束自身可以是泛型類(lèi)型,如下所示:

C# 類(lèi)型參數(shù)約束代碼

 
 
 
  1. class EmployeeList  where T : Employee, IEmployee, System.IComparable ,  new()  
  2. {  
  3.     // ...  

通過(guò)約束類(lèi)型參數(shù),可以增加約束類(lèi)型及其繼承層次結(jié)構(gòu)中的所有類(lèi)型所支持的允許操作和方法調(diào)用的數(shù)量。因此,在設(shè)計(jì)泛型類(lèi)或方法時(shí),如果要對(duì)泛型成員執(zhí)行除簡(jiǎn)單賦值之外的任何操作或調(diào)用 System.Object 不支持的任何方法,您將需要對(duì)該類(lèi)型參數(shù)應(yīng)用約束。

在應(yīng)用 where T : class 約束時(shí),建議不要對(duì)類(lèi)型參數(shù)使用 == 和 != 運(yùn)算符,因?yàn)檫@些運(yùn)算符僅測(cè)試引用同一性而不測(cè)試值相等性。即使在用作參數(shù)的類(lèi)型中重載這些運(yùn)算符也是如此。下面的代碼說(shuō)明了這一點(diǎn);即使 String 類(lèi)重載 == 運(yùn)算符,輸出也為 false。

C# 類(lèi)型參數(shù)約束代碼

 
 
 
  1. public static void OpTest (T s, T t) where T :  class 
  2. {  
  3.     System.Console.WriteLine(s == t);  
  4. }  
  5. static void Main()  
  6. {  
  7.     string s1 = "foo";  
  8.     System.Text.StringBuilder sb = new System.Text.StringBuilder("foo");  
  9.     string s2 = sb.ToString();  
  10.     OpTest(s1, s2);  

這種情況的原因在于,編譯器在編譯時(shí)僅知道 T 是引用類(lèi)型,因此必須使用對(duì)所有引用類(lèi)型都有效的默認(rèn)運(yùn)算符。如果需要測(cè)試值相等性,建議的方法是同時(shí)應(yīng)用 where T : IComparable 約束,并在將用于構(gòu)造泛型類(lèi)的任何類(lèi)中實(shí)現(xiàn)該接口。

C# 未綁定的類(lèi)型參數(shù)

沒(méi)有約束的類(lèi)型參數(shù)(如公共類(lèi) SampleClass {} 中的 T)稱(chēng)為未綁定的類(lèi)型參數(shù)。未綁定的類(lèi)型參數(shù)具有以下規(guī)則:

不能使用 != 和 == 運(yùn)算符,因?yàn)闊o(wú)法保證具體類(lèi)型參數(shù)能支持這些運(yùn)算符。

可以在它們與 System.Object 之間來(lái)回轉(zhuǎn)換,或?qū)⑺鼈冿@式轉(zhuǎn)換為任何接口類(lèi)型。

可以將它們與 null 進(jìn)行比較。將未綁定的參數(shù)與 null 進(jìn)行比較時(shí),如果類(lèi)型參數(shù)為值類(lèi)型,則該比較將始終返回 false。

C# 裸類(lèi)型約束

用作約束的泛型類(lèi)型參數(shù)稱(chēng)為裸類(lèi)型約束。當(dāng)具有自己的類(lèi)型參數(shù)的成員函數(shù)需要將該參數(shù)約束為包含類(lèi)型的類(lèi)型參數(shù)時(shí),裸類(lèi)型約束很有用,如下面的示例所示:

C# 類(lèi)型參數(shù)約束代碼

 
 
 
  1. class List  
  2. {  
  3.     void Add(List items) where U : T {/*...*/}  

在上面的示例中,T 在 Add 方法的上下文中是一個(gè)裸類(lèi)型約束,而在 List 類(lèi)的上下文中是一個(gè)未綁定的類(lèi)型參數(shù)。

裸類(lèi)型約束還可以在泛型類(lèi)定義中使用。注意,還必須已經(jīng)和其他任何類(lèi)型參數(shù)一起在尖括號(hào)中聲明了裸類(lèi)型約束:

C# 類(lèi)型參數(shù)約束代碼

 
 
 
  1. //naked type constraint  
  2. public class SampleClass  where T : V { } 

泛型類(lèi)的裸類(lèi)型約束的作用非常有限,因?yàn)榫幾g器除了假設(shè)某個(gè)裸類(lèi)型約束派生自 System.Object 以外,不會(huì)做其他任何假設(shè)。在希望強(qiáng)制兩個(gè)類(lèi)型參數(shù)之間的繼承關(guān)系的情況下,可對(duì)泛型類(lèi)使用裸類(lèi)型約束。

C# 類(lèi)型參數(shù)約束的相關(guān)內(nèi)容就向你介紹到這里,希望對(duì)你了解和學(xué)習(xí)C# 類(lèi)型參數(shù)約束有所幫助。


分享文章:C#類(lèi)型參數(shù)約束分析及應(yīng)用淺析
文章轉(zhuǎn)載:http://www.5511xx.com/article/ccccjji.html