C# 之泛型詳解

這篇文章主要來講講c#中的泛型,因爲泛型在c#中有很重要的位置,對於寫出高可讀性,高性能的代碼有着關鍵的作用。

一、什麼是泛型?

泛型是 2.0 版 C# 語言和公共語言運行庫 (CLR) 中的一個非常重要的新功能。

我們在編程程序時,經常會遇到功能非常相似的模塊,只是它們處理的數據不一樣。但我們沒有辦法,只能分別寫多個方法來處理不同的數據類型。這個時候,那麼問題來了,有沒有一種辦法,用同一個方法來處理傳入不同種類型參數的辦法呢?泛型的出現就是專門來解決這個問題的,可以看出,微軟還是很貼心的。

二、爲什麼要使用泛型?

接下來我們來看一段代碼。

複製代碼
public class GenericClass
    {
        public void ShowInt(int n)
        {
            Console.WriteLine("ShowInt print {0},ShowInt Parament Type Is {1}",n,n.GetType());
        }
        public void ShowDateTime(DateTime dt)
        {
            Console.WriteLine("ShowDateTime print {0},ShowDateTime Parament Type Is {1}", dt, dt.GetType());
        }
        public void ShowPeople(People people)
        {
            Console.WriteLine("ShowPeople print {0},ShowPeople Parament Type Is {1}", people, people.GetType());
        }
    }
複製代碼
複製代碼
static void Main(string[] args)
        {
            GenericClass generice = new GenericClass();
            generice.ShowInt(11);
            generice.ShowDateTime(DateTime.Now);
            generice.ShowPeople(new People { Id = 11, Name = "Tom" });

            Console.ReadKey();
        }
複製代碼

顯示結果:

我們可以看出這三個方法,除了傳入的參數不同外,其裏面實現的功能都是一樣的。在1.1版的時候,還沒有泛型這個概念,那麼怎麼辦呢。就有人想到了OOP三大特性之一的繼承,我們知道,C#語言中,所有類型都源自同一個類型,那就是object。

複製代碼
public class GenericClass
    {
        public void ShowObj(object obj)
        {
            Console.WriteLine("ShowObj print {0},ShowObj Parament Type Is {1}", obj, obj.GetType());
        }
    }
        static void Main(string[] args)
        {
            Console.WriteLine("*****************object調用*********************");
            generice.ShowObj(11);
            generice.ShowObj(DateTime.Now);
            generice.ShowObj(new People { Id = 11, Name = "Tom" });

            Console.ReadKey();
        }
複製代碼

顯示結果:

我們可以看出,目地是達到了。解決了代碼的可讀性,但是這樣又有個不好的地方了,我們這樣做實際上是一個裝箱拆箱操作,會損耗性能。

終於,微軟在2.0的時候發佈了泛型。接下來我們用泛型方法來實現該功能。

三、泛型類型參數

在使用泛型方法之前,我們先來了解下有關於泛型的一些知識。

在泛型類型或方法定義中,類型參數是在其實例化泛型類型的一個變量時,客戶端指定的特定類型的佔位符。 泛型類( GenericList<T>)無法按原樣使用,因爲它不是真正的類型;它更像是類型的藍圖。 若要使用 GenericList<T>,客戶端代碼必須通過指定尖括號內的類型參數來聲明並實例化構造類型。 此特定類的類型參數可以是編譯器可識別的任何類型。 可創建任意數量的構造類型實例,其中每個使用不同的類型參數,如下所示:

GenericList<float> list1 = new GenericList<float>();
GenericList<ExampleClass> list2 = new GenericList<ExampleClass>();
GenericList<ExampleStruct> list3 = new GenericList<ExampleStruct>();

在 GenericList<T> 的每個實例中,類中出現的每個 T 在運行時均會被替換爲類型參數。 通過這種替換,我們已通過使用單個類定義創建了三個單獨的類型安全的有效對象。 

三、泛型約束

定義泛型類時,可以對客戶端代碼能夠在實例化類時用於類型參數的幾種類型施加限制。 如果客戶端代碼嘗試使用約束所不允許的類型來實例化類,則會產生編譯時錯誤。 這些限制稱爲約束。 通過使用 where 上下文關鍵字指定約束。 下表列出了六種類型的約束:

where T:結構(類型參數必須是值類型。可以指定除 Nullable 以外的任何值類型。)

複製代碼
class MyClass<U>
        where U : struct///約束U參數必須爲“值 類型”
 { }

 public void MyMetod<T>(T t)
       where T : struct
 {          
 }
複製代碼

where T:類(類型參數必須是引用類型;這一點也適用於任何類、接口、委託或數組類型。)

複製代碼
class MyClass<U>
        where U : class///約束U參數必須爲“引用類型”
 { }

 public void MyMetod<T>(T t)
       where T : class
 {          
 }
複製代碼

where T:new()(類型參數必須具有無參數的公共構造函數。當與其他約束一起使用時,new() 約束必須最後指定。)

class EmployeeList<T> where T : Employee, IEmployee, System.IComparable<T>, new()
{
    // ...
}

where T:<基類名>(類型參數必須是指定的基類或派生自指定的基類。)

public class Employee{}

public class GenericList<T> where T : Employee

where T:<接口名稱>(類型參數必須是指定的接口或實現指定的接口。可以指定多個接口約束。約束接口也可以是泛型的。)

複製代碼
/// <summary>
    /// 接口
    /// </summary>
    interface IMyInterface
    {
    }

    /// <summary>
    /// 定義的一個字典類型
    /// </summary>
    /// <typeparam name="TKey"></typeparam>
    /// <typeparam name="TVal"></typeparam>
    class Dictionary<TKey, TVal>
        where TKey : IComparable, IEnumerable
        where TVal : IMyInterface
    {
        public void Add(TKey key, TVal val)
        {
        }
    }
複製代碼

where T:U(爲 T 提供的類型參數必須是爲 U 提供的參數或派生自爲 U 提供的參數。也就是說T和U的參數必須一樣)

class List<T>
{
    void Add<U>(List<U> items) where U : T {/*...*/}
}

以上就是對六種泛型的簡單示例,當然泛型約束不僅僅適用於類,接口,對於泛型方法,泛型委託都同樣適用。

三、泛型方法

複製代碼
public class GenericClass
    {
        public void ShowT<T>(T t)
        {
            Console.WriteLine("ShowT print {0},ShowT Parament Type Is {1}", t, t.GetType());
        }
    }
static void Main(string[] args)
        {
            Console.WriteLine("*****************泛型方法調用*********************");
            generice.ShowT<int>(11);
            generice.ShowT<DateTime>(DateTime.Now);
            generice.ShowT<People>(new People { Id = 11, Name = "Tom" });

            Console.ReadKey();
        }
複製代碼

顯示結果:

也是一樣的,現在終於實現了我們想要達到的效果了。我們可以看出,無論是什麼方式調用,最後我們獲取出來的類型都是原始類型。我們知道,用object獲取是利用了繼承這一特性,當編譯器編譯的時候,我們傳入的參數會進行裝箱操作,當我們獲取的時候又要進行拆箱操作,這個方法會損耗性能 。那麼泛型方法實現的原理又是怎樣的呢?首先,我們要知道,泛型是一個語法糖,在我們調用泛型方法,編譯器進行編譯時,纔會確定傳入的參數的類型,從而生成副本方法。這個副本方法與原始方法一法,所以不會有裝箱拆箱操作,也就沒有損耗性能這回事了。

四、泛型類

泛型類封裝不特定於特定數據類型的操作。

通常,創建泛型類是從現有具體類開始,然後每次逐個將類型更改爲類型參數,直到泛化和可用性達到最佳平衡。

創建自己的泛型類時,需要考慮以下重要注意事項:

  • 要將哪些類型泛化爲類型參數。

               通常,可參數化的類型越多,代碼就越靈活、其可重用性就越高。 但過度泛化會造成其他開發人員難以閱讀或理解代碼。

  • 要將何種約束(如有)應用到類型參數

 

         其中一個有用的規則是,應用最大程度的約束,同時仍可處理必須處理的類型。 例如,如果知道泛型類僅用於引用類型,則請應用類約束。 這可防止將類意外用於值類型,並     使你可在   T  上使用  as  運算符和檢查 null 值。      

 

  • 是否將泛型行爲分解爲基類和子類。

    因爲泛型類可用作基類,所以非泛型類的相同設計注意事項在此也適用。 請參閱本主題後文有關從泛型基類繼承的規則。

  • 實現一個泛型接口還是多個泛型接口。

  • 複製代碼
    class BaseNode { }
    class BaseNodeGeneric<T> { }
    
    // concrete type
    class NodeConcrete<T> : BaseNode { }
    
    //closed constructed type
    class NodeClosed<T> : BaseNodeGeneric<int> { }
    
    //open constructed type 
    class NodeOpen<T> : BaseNodeGeneric<T> { }
    複製代碼

    五、泛型接口

    • 定義一個泛型接口:
    • interface IMyGenericInterface<T>
      {
      }
      • 一個接口可定義多個類型參數,如下所示:
      • interface IMyGenericInterface<TKey,TValue>
        {
        }
        • 具體類可實現封閉式構造接口,如下所示:
        • interface IBaseInterface<T> { }
          
          class SampleClass : IBaseInterface<string> { }//如果T有約束,那麼string類型必須得滿足T的約束

六、泛型委託

委託可以定義它自己的類型參數。 引用泛型委託的代碼可以指定類型參數以創建封閉式構造類型,就像實例化泛型類或調用泛型方法一樣,如以下示例中所示:

複製代碼
class Program
    {
        static void Main(string[] args)
        {
            Del<int> m1 = new Del<int>(Notify);
            m1.Invoke(1111);
            Del<string> m2 = new Del<string>(Notify);
            m2.Invoke("字符串");

            Console.ReadKey();
        }

        public delegate void Del<T>(T item);
        public static void Notify(int i) { Console.WriteLine("{0} type is {1}", i,i.GetType()); }
        public static void Notify(string str) { Console.WriteLine("{0} type is {1}", str, str.GetType()); }
       
    }
複製代碼

運行結果:

七、泛型代碼中的默認關鍵字:Default

在泛型類和泛型方法中產生的一個問題是,在預先未知以下情況時,如何將默認值分配給參數化類型 T:

  • T 是引用類型還是值類型。

  • 如果 T 爲值類型,則它是數值還是結構。

給定參數化類型 T 的一個變量 t,只有當 T 爲引用類型時,語句 t = null 纔有效;只有當 T 爲數值類型而不是結構時,語句 t = 0 才能正常使用。解決方案是使用 default 關鍵字,此關鍵字對於引用類型會返回空,對於數值類型會返回零。對於結構,此關鍵字將返回初始化爲零或空的每個結構成員,具體取決於這些結構是值類型還是引用類型。

複製代碼
namespace MyGeneric
{
    class Program
    {
        static void Main(string[] args)
        {
            object obj1=GenericToDefault<string>();
            object obj2 = GenericToDefault<int>();
            object obj3 = GenericToDefault<StructDemo>();
            Console.ReadKey();
        }
        public static T GenericToDefault<T>() 
        {
            return default(T);
        }
    }
    public struct StructDemo
    {
        public int Id { get; set; }
        public string Name { get; set; }
    }
}
複製代碼

運行結果:

 

 

  原文鏈接:https://www.cnblogs.com/hhzblogs/p/7820005.html

發佈了18 篇原創文章 · 獲贊 26 · 訪問量 5萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章