欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C# 泛型的簡(jiǎn)單理解(安全、集合、方法、約束、繼承)分享

 更新時(shí)間:2013年10月24日 15:28:45   作者:  
這篇文章介紹了C# 泛型的簡(jiǎn)單理解(安全、集合、方法、約束、繼承),有需要的朋友可以參考一下

前言

泛型允許你在編譯時(shí)實(shí)現(xiàn)類型安全。它們?cè)试S你創(chuàng)建一個(gè)數(shù)據(jù)結(jié)構(gòu)而不限于一特定的數(shù)據(jù)類型。然而,當(dāng)使用該數(shù)據(jù)結(jié)構(gòu)時(shí),編譯器保證它使用的類型與類型安全是相一致的。泛型提供了類型安全,但是沒(méi)有造成任何性能損失和代碼臃腫。在這方面,它們很類似于C++中的模板,不過(guò)它們?cè)趯?shí)現(xiàn)上是很不同的。

使用泛型集合

 .NET 2.0的System.Collections.Generics 命名空間包含了泛型集合定義。各種不同的集合/容器類都被"參數(shù)化"了。為使用它們,只需簡(jiǎn)單地指定參數(shù)化的類型即可。

復(fù)制代碼 代碼如下:

ArrayList array = new ArrayList();
            array.Add(3);
            array.Add(4);
            array.Add(5.0);
            int total = 0;
            foreach (int val in array)
            {
                total = total + val;
            }
            Console.WriteLine("Total is {0}", total);

這段代碼編譯肯定沒(méi)問(wèn)題的,不過(guò)在運(yùn)行的時(shí)候就會(huì)報(bào)錯(cuò)。因?yàn)樵趂oreach哪里定義的都是int,而在添加的是5.0很明顯是個(gè)double類型的。


復(fù)制代碼 代碼如下:

List<int> aList = new List<int>();
            aList.Add(3);
            aList.Add(4);
            //aList.Add(5.0);
            int totalList = 0;
            foreach(int val in aList)
            {
                totalList = totalList + val;
            }
            Console.WriteLine("Total is {0}", totalList);

這段代碼其實(shí)也沒(méi)什么問(wèn)題,如果把注釋的哪一行的注釋去掉,那么在編譯的時(shí)候就直接報(bào)錯(cuò)了,因?yàn)榫幾g器指出它不能發(fā)送值5.0到方法Add(),因?yàn)樵摲椒▋H接受int型。

不同于ArrayList,這里的代碼實(shí)現(xiàn)了類型安全。

CLR對(duì)于泛型的支持

 泛型不僅是一個(gè)語(yǔ)言級(jí)上的特征。.NET CLR能識(shí)別出泛型。在這種意義上說(shuō),泛型的使用是.NET中最為優(yōu)秀的特征之一。對(duì)每個(gè)用于泛型化的類型的參數(shù),類也同樣沒(méi)有脫離開微軟中間語(yǔ)言(MSIL)。換句話說(shuō),你的配件集僅包含你的參數(shù)化的數(shù)據(jù)結(jié)構(gòu)或類的一個(gè)定義,而不管使用多少種不同的類型來(lái)表達(dá)該參數(shù)化的類型。例如,如果你定義一個(gè)泛型類型MyList<T>,僅僅該類型的一個(gè)定義出現(xiàn)在MSIL中。當(dāng)程序執(zhí)行時(shí),不同的類被動(dòng)態(tài)地創(chuàng)建,每個(gè)類對(duì)應(yīng)該參數(shù)化類型的一種類型。如果你使用MyList<int>和MyList<double>,有兩種類即被創(chuàng)建。

接下來(lái)創(chuàng)建一個(gè)簡(jiǎn)單的泛型類

復(fù)制代碼 代碼如下:

public class MyList<T>
    {
        private static int objCount = 0;
        public  MyList()
        {
            objCount++;
        }

        public int Count
        {
            get { return objCount; }
        }
    }

該例中,我創(chuàng)建了一個(gè)稱為MyList泛型類。為把它參數(shù)化,我簡(jiǎn)單地插入了一個(gè)尖括號(hào)。在<>內(nèi)的T代表了實(shí)際的當(dāng)使用該類時(shí)要指定的類型。在MyList類中,定義了一個(gè)靜態(tài)字段objCount。我在構(gòu)造器中增加它的值。因此我能發(fā)現(xiàn)使用我的類的用戶共創(chuàng)建了多少個(gè)那種類型的對(duì)象。屬性Count返回與被調(diào)用的實(shí)例同類型的實(shí)例的數(shù)目。

復(fù)制代碼 代碼如下:

public class SampleClass
    {

    }

    class Program
    {
        static void Main(string[] args)
        {
            MyList<int> myIntList=new MyList<int>();
            MyList<int> myIntList2=new MyList<int>();
            MyList<double> myDoubleList=new MyList<double>();
            MyList<SampleClass> mySampleList=new MyList<SampleClass>();

            Console.WriteLine(myIntList.Count);
            Console.WriteLine(myIntList2.Count);
            Console.WriteLine(myDoubleList.Count);
            Console.WriteLine(mySampleList.Count);
            Console.WriteLine(new MyList<SampleClass>().Count);
            Console.ReadLine();
        }
    }

在Main()方法,我創(chuàng)建了MyList<int>的兩個(gè)實(shí)例,一個(gè)MyList<double>的實(shí)例,還有兩個(gè)MyList<SampleClass>的實(shí)例--其中SampleClass是我已定義了的類。問(wèn)題是:Count(上面的程序的輸出)的值該是多少?在你繼閱讀之前,試一試回答這個(gè)問(wèn)題。

前面兩個(gè)2對(duì)應(yīng)MyList<int>,第一個(gè)1對(duì)應(yīng)MyList<double>,第二個(gè)1對(duì)應(yīng)MyList<SampleClass>--在此,僅創(chuàng)建一個(gè)這種類型的實(shí)例。最后一個(gè)2對(duì)應(yīng)MyList<SampleClass>,因?yàn)榇a中又創(chuàng)建了這種類型的另外一個(gè)實(shí)例。上面的例子說(shuō)明MyList<int>是一個(gè)與MyList<double>不同的類,而MyList<double>又是一個(gè)與MyList<SampleClass>不同的類。因此,在這個(gè)例中,我們有四個(gè)類:MyList: MyList<T>,MyList<int>,MyList<double>和MyList<X>。注意,雖然有4個(gè)MyList類,但僅有一個(gè)被存儲(chǔ)在MSIL。怎么能證明這一點(diǎn)?請(qǐng)看下圖顯示出的使用工具ildasm.exe生成的MSIL代碼。

泛型方法

 除了有泛型類,你也可以有泛型方法。泛型方法可以是任何類的一部分。

復(fù)制代碼 代碼如下:

public static void Copy<T>(List<T> source, List<T> destination)
        {
            foreach (T obj in source)
            {
                destination.Add(obj);
            }
        }
        static void Main(string[] args)
        {
            List<int> lst1 = new List<int>();
            lst1.Add(2);
            lst1.Add(4);
            List<int> lst2 = new List<int>();
            Copy(lst1, lst2);
            Console.WriteLine(lst2.Count);
            Console.ReadLine();
        }

Copy()方法就是一個(gè)泛型方法,它與參數(shù)化的類型T一起工作。當(dāng)在Main()中激活Copy()時(shí),編譯器根據(jù)提供給Copy()方法的參數(shù)確定出要使用的具體類型。

約束機(jī)制及其優(yōu)點(diǎn)

 一個(gè)泛型類允許你寫自己的類而不必拘泥于任何類型,但允許你的類的使用者以后可以指定要使用的具體類型。通過(guò)對(duì)可能會(huì)用于參數(shù)化的類型的類型施加約束,這給你的編程帶來(lái)很大的靈活性--你可以控制建立你自己的類。讓我們分析一個(gè)例子:

復(fù)制代碼 代碼如下:

        public static T Max<T>(T op1, T op2)
        {
            if (op1.CompareTo(op2) < 0)
                 return op1;
            return op2;
        }

編譯代碼將會(huì)有一個(gè)錯(cuò)誤。

假定我需要這種類型以支持CompareTo()方法的實(shí)現(xiàn)。我能夠通過(guò)加以約束--為參數(shù)化類型指定的類型必須要實(shí)現(xiàn)IComparable接口--來(lái)指定這一點(diǎn)。

復(fù)制代碼 代碼如下:

        public static T Max<T>(T op1, T op2)where T:IComparable
        {
            if (op1.CompareTo(op2) < 0)
                 return op1;
            return op2;
        }

好了,我指定的約束是,用于參數(shù)化類型的類型必須繼承自(實(shí)現(xiàn))Icomparable。現(xiàn)在可以編譯成功,并且調(diào)用了。

下面的約束是可以使用的:

  where T : struct 類型必須是一種值類型(struct)

  where T : class 類型必須是一種引用類型(class)

  where T : new() 類型必須有一個(gè)無(wú)參數(shù)的構(gòu)造器

  where T : class_name 類型可以是class_name或者是它的一個(gè)子類

  where T : interface_name 類型必須實(shí)現(xiàn)指定的接口

  你可以指定約束的組合,就象: where T : IComparable, new()。這就是說(shuō),用于參數(shù)化類型的類型必須實(shí)現(xiàn)Icomparable接口并且必須有一個(gè)無(wú)參構(gòu)造器。

繼承與泛型

一個(gè)使用參數(shù)化類型的泛型類,象MyClass1<T>,稱作開放結(jié)構(gòu)的泛型。一個(gè)不使用參數(shù)化類型的泛型類,象MyClass1<int>,稱作封閉結(jié)構(gòu)的泛型。

你可以從一個(gè)封閉結(jié)構(gòu)的泛型進(jìn)行派生;也就是說(shuō),你可以從另外一個(gè)稱為MyClass1的類派生一個(gè)稱為MyClass2的類,就象:

復(fù)制代碼 代碼如下:

public class MyClass2<T> : MyClass1<int>

你也可以從一個(gè)開放結(jié)構(gòu)的泛型進(jìn)行派生,如果類型被參數(shù)化的話,如:
復(fù)制代碼 代碼如下:

public class MyClass2<T> : MyClass2<T>

是有效的,但是
復(fù)制代碼 代碼如下:

public class MyClass2<T> : MyClass2<Y>

是無(wú)效的,這里Y是一個(gè)被參數(shù)化的類型。非泛型類可以從一個(gè)封閉結(jié)構(gòu)的泛型類進(jìn)行派生,但是不能從一個(gè)開放結(jié)構(gòu)的泛型類派生。
復(fù)制代碼 代碼如下:

public class MyClass : MyClass1<int>

是有效的, 但是
復(fù)制代碼 代碼如下:

public class MyClass : MyClass1<T>

是無(wú)效的。

相關(guān)文章

  • C# 元組和值元組的具體使用

    C# 元組和值元組的具體使用

    這篇文章主要介紹了C# 元組和值元組的具體使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10
  • C#中HttpWebRequest的用法詳解

    C#中HttpWebRequest的用法詳解

    這篇文章主要介紹了C#中HttpWebRequest的用法,以實(shí)例的形式詳細(xì)敘述了HttpWebRequest類中GET與POST的用法,非常具有參考借鑒價(jià)值,需要的朋友可以參考下
    2014-11-11
  • C#?BitArray(點(diǎn)矩陣)轉(zhuǎn)換成int和string的方法實(shí)現(xiàn)

    C#?BitArray(點(diǎn)矩陣)轉(zhuǎn)換成int和string的方法實(shí)現(xiàn)

    BitArray?類管理一個(gè)緊湊型的位值數(shù)組,它使用布爾值來(lái)表示,本文主要介紹了C#?BitArray(點(diǎn)矩陣)轉(zhuǎn)換成int和string的方法實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2022-05-05
  • C#中Timer實(shí)現(xiàn)Tick使用精度的問(wèn)題

    C#中Timer實(shí)現(xiàn)Tick使用精度的問(wèn)題

    這篇文章主要介紹了C#中Timer實(shí)現(xiàn)Tick使用精度的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • WPF實(shí)現(xiàn)手風(fēng)琴式輪播圖切換效果

    WPF實(shí)現(xiàn)手風(fēng)琴式輪播圖切換效果

    這篇文章主要為大家詳細(xì)介紹了WPF實(shí)現(xiàn)手風(fēng)琴式輪播圖切換效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-09-09
  • C#語(yǔ)法之泛型的多種應(yīng)用

    C#語(yǔ)法之泛型的多種應(yīng)用

    這篇文章主要介紹了C#語(yǔ)法之泛型的多種應(yīng)用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-07-07
  • C#設(shè)置與獲取環(huán)境變量的方法詳解

    C#設(shè)置與獲取環(huán)境變量的方法詳解

    這篇文章主要給大家介紹了關(guān)于C#設(shè)置與獲取環(huán)境變量的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。
    2018-03-03
  • .NET操作NPOI實(shí)現(xiàn)Excel的導(dǎo)入導(dǎo)出

    .NET操作NPOI實(shí)現(xiàn)Excel的導(dǎo)入導(dǎo)出

    NPOI是指構(gòu)建在POI 3.x版本之上的一個(gè)程序,NPOI可以在沒(méi)有安裝Office的情況下對(duì)Word或Excel文檔進(jìn)行讀寫操作,下面小編為大家介紹了如何操作NPOI實(shí)現(xiàn)Excel的導(dǎo)入導(dǎo)出,需要的可以參考一下
    2023-09-09
  • C#中Hash table的一些操作方法講解

    C#中Hash table的一些操作方法講解

    今天小編就為大家分享一篇關(guān)于C#中Hash table的一些操作方法講解,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2019-01-01
  • C#中三種Timer計(jì)時(shí)器的詳細(xì)用法

    C#中三種Timer計(jì)時(shí)器的詳細(xì)用法

    這篇文章介紹了C#中三種Timer計(jì)時(shí)器的詳細(xì)用法,文中通過(guò)示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2022-05-05

最新評(píng)論