码迷,mamicode.com
首页 > 其他好文 > 详细

理解泛型

时间:2015-06-24 18:49:08      阅读:111      评论:0      收藏:0      [点我收藏+]

标签:

1.1 理解泛型
1.1.1 为什么要有泛型?

我想不论大家通过什么方式进入了计算机程序设计这个行业,都免不了

要面对数据结构和算法这个话题。因为它是计算机科学的一门基础学科

,往往越是底层的部分,对于数据结构或者算法的时间效率和空间效率

的要求就越高。比如说,当你在一个集合类型(例如ArrayList)的实例

上调用Sort()方法对它进行排序时,.Net框架在底层就应用了快速排序

算法。.Net框架中快速排序方法名称叫QuickSort(),它位于Array类型

中,这可以通过Reflector.exe工具查看到。

我们现在并不是要讨论这个QuickSort()实现的好不好,效率高还是不高

,这偏离了我们的主题。但是我想请大家思考一个问题:如果由你来实

现一个排序算法,你会怎么做?好吧,我们把题目限定得再窄一些,我

们来实现一个最简单的冒泡排序(Bubble Sort)算法,如果你没有使用

泛型的经验,我猜测你可能会毫不犹豫地写出下面的代码来,因为这是

大学教程的标准实现:

public class SortHelper{
    public void BubbleSort(int[] array) {
        int length = array.Length;

        for (int i = 0; i <= length - 2; i++) {
            for (int j = length - 1; j >= 1; j--) {

                // 对两个元素进行交换
                if (array[j] < array[j - 1] ) {
                    int temp = array[j];
                    array[j] = array[j - 1];
                    array[j - 1] = temp;    
                }
            }
        }
    }
}

对冒泡排序不熟悉的读者,可以放心地忽略上面代码的方法体,它不会

对你理解泛型造成丝毫的障碍,你只要知道它所实现的功能就可以了:

将一个数组的元素按照从小到大的顺序重新排列。我们对这个程序进行

一个小小地测试:

class Program {
    static void Main(string[] args) {
        SortHelper sorter = new SortHelper();
        
        int[] array = { 8, 1, 4, 7, 3 };

        sorter.BubbleSort(array);

        foreach(int i in array){
            Console.Write("{0} ", i);
        }

        Console.WriteLine();
        Console.ReadKey();
    }
}

输出为:

1 3 4 7 8

我们发现它工作良好,欣喜地认为这便是最好的解决方案了。直到不久

之后,我们需要对一个byte类型的数组进行排序,而我们上面的排序算

法只能接受一个int类型的数组,尽管我们知道它们是完全兼容的,因为

byte类型是int类型的一个子集,但C#是一个强类型的语言,我们无法在

一个接受int数组类型的地方传入一个byte数组。好吧,没有关系,现在

看来唯一的办法就是将代码复制一遍,然后将方法的签名改一个改了:

public class SortHelper {
    public void BubbleSort(int[] array) {
        int length = array.Length;

        for (int i = 0; i <= length - 2; i++) {
            for (int j = length - 1; j >= 1; j--) {

                // 对两个元素进行交换
                if (array[j] < array[j - 1]) {
                    int temp = array[j];
                    array[j] = array[j - 1];
                    array[j - 1] = temp;
                }
            }
        }
    }


    public void BubbleSort(byte[] array) {
        int length = array.Length;

        for (int i = 0; i <= length - 2; i++) {
            for (int j = length - 1; j >= 1; j--) {

                // 对两个元素进行交换
                if (array[j] < array[j - 1]) {
                    int temp = array[j];
                    array[j] = array[j - 1];
                    array[j - 1] = temp;
                }
            }
        }
    }
}

OK,我们再一次解决了问题,尽管总觉得哪里有点别扭,但是这段代码

已经能够工作,按照敏捷软件开发的思想,不要过早地进行抽象和应对

变化,当变化第一次出现时,使用最快的方法解决它,当变化第二次出

现时,再进行更好的构架和设计。这样做的目的是为了避免过度设计,

因为很有可能第二次变化永远也不会出现,而你却花费了大量的时间精

力制造了一个永远也用不到的“完美设计”。这很像一个谚语,“fool

me once,shame on you. fool me twice, shame on me.”,翻译过

来的意思是“愚弄我一次,是你坏;愚弄我两次,是我蠢”。

美好的事情总是很难长久,我们很快需要对一个char类型的数组进行排

序,我们当然可以仿照byte类型数组的作法,继续采用复制粘贴大法,

然后修改一下方法的签名。但是很遗憾,我们不想让它愚弄我们两次,

因为谁也不想证明自己很蠢,所以现在是时候思考一个更佳的解决方案

了。

我们仔细地对比这两个方法,会发现这两个方法的实现完全一样,除了

方法的签名不同以外,没有任何的区别。如果你曾经开发过Web站点程序

,会知道对于一些浏览量非常大的站点,为了避免服务器负担过重,通

常会采用静态页面生成的方式,因为使用Url重写仍要要耗费大量的服务

器资源,但是生成为html静态网页后,服务器仅仅是返回客户端请求的

文件,能够极大的减轻服务器负担。

在Web上实现过静态页面生成时,有一种常用的方法,就是模板生成法,

它的具体作法是:每次生成静态页面时,先加载模板,模板中含有一些

用特殊字符标记的占位符,然后我们从数据库读取数据,使用读出的数

据将模板中的占位符替换掉,最后将模板按照一定的命名规则在服务器

上保存成静态的html文件。

我们发现这里的情况是类似的,我来对它进行一个类比:我们将上面的

方法体视为一个模板,将它的方法签名视为一个占位符,因为它是一个

占位符,所以它可以代表任何的类型,这和静态页面生成时模板的占位

符可以用来代表来自数据库中的任何数据道理是一样的。接下来就是定

义占位符了,我们再来审视一下这三个方法的签名:

public void BubbleSort(int[] array)
public void BubbleSort(byte[] array)
public void BubbleSort(char[] array)

会发现定义占位符的最好方式就是将int[]、byte[]、char[]用占位符替

代掉,我们管这个占位符用T[]来表示,其中T可以代表任何类型,这样

就屏蔽了三个方法签名的差异:

public void BubbleSort(T[] array) {
    int length = array.Length;

    for (int i = 0; i <= length - 2; i++) {
        for (int j = length - 1; j >= 1; j--) {

            // 对两个元素进行交换
            if (array[j] < array[j - 1]) {
                T temp = array[j];
                array[j] = array[j - 1];
                array[j - 1] = temp;
            }
        }
    }
}

现在看起来清爽多了,但是我们又发现了一个问题:当我们定义一个类

,而这个类需要引用它本身以外的其他类型时,我们可以定义有参数的

构造函数,然后将它需要的参数从构造函数传进来。但是在上面,我们

的参数T本身就是一个类型(类似于int、byte、char,而不是类型的实

例,比如1和‘a‘)。很显然我们无法在构造函数中传递这个T类型的数组

,因为参数都是出现在类型实例的位置,而T是类型本身,它的位置不对

。比如下面是通常的构造函数:

public SortHelper(类型 类型实例名称);

而我们期望的构造函数函数是:

public SortHelper(类型);

此时就需要使用一种特殊的语法来传递这个T占位符,不如我们定义这样

一种语法来传递吧:

public class SortHelper<T> {
    public void BubbleSort(T[] array){
        // 方法实现体
    }
}

我们在类名称的后面加了一个尖括号,使用这个尖括号来传递我们的占

位符,也就是类型参数。接下来,我们来看看如何来使用它,当我们需

要为一个int类型的数组排序时:

SortHelper<int> sorter = new SortHelper<int>();
int[] array = { 8, 1, 4, 7, 3 };
sorter.BubbleSort(array);

当我们需要为一个byte类型的数组排序时:

SortHelper<byte> sorter = new SortHelper<byte>();
byte [] array = { 8, 1, 4, 7, 3 };
sorter.BubbleSort(array);

相信你已经发觉,其实上面所做的一切实现了一个泛型类。这是泛型的

一个最典型的应用,可以看到,通过使用泛型,我们极大地减少了重复

代码,使我们的程序更加清爽,泛型类就类似于一个模板,可以在需要

时为这个模板传入任何我们需要的类型。

我们现在更专业一些,为这一节的占位符起一个正式的名称,在.Net中

,它叫做类型参数 (Type Parameter),下面一小节,我们将学习类型

参数约束。
1.1.2 类型参数约束

实际上,如果你运行一下上面的代码就会发现它连编译都通过不了,为

什么呢?考虑这样一个问题,假如我们自定义一个类型,它定义了书,

名字叫做Book,它含有两个字段:一个是int类型的Id,是书的标识符;

一个是string类型的Title,代表书的标题。因为我们这里是一个范例,

为了既能说明问题又不偏离主题,所以这个Book类型只含有这两个字段



public class Book {
    private int id;
    private string title;

    public Book() { }

    public Book(int id, string title) {
        this.id = id;
        this.title = title;
    }

    public int Id {
        get { return id; }
        set { id = value; }
    }

    public string Title {
        get { return title; }
        set { title = value; }
    }
}

现在,我们创建一个Book类型的数组,然后试着使用上一小节定义的泛

型类来对它进行排序,我想代码应该是这样子的:

Book[] bookArray = new Book[2];

Book book1 = new Book(124, ".Net之美");
Book book2 = new Book(45, "C# 3.0揭秘");

bookArray[0] = book1;
bookArray[1] = book2;

SortHelper<Book> sorter = new SortHelper<Book>();
sorter.BubbleSort(bookArray);

foreach (Book b in bookArray) {
    Console.WriteLine("Id:{0}", b.Id);
    Console.WriteLine("Title:{0}\n", b.Title);
}

可能现在你还是没有看到会有什么问题,你觉得上一节的代码很通用,

那么让我们看得再仔细一点,再看一看SortHelper类的BubbleSort()方

法的实现吧,为了避免你回头再去翻上一节的代码,我将它复制了下来



public void BubbleSort(T[] array) {
    int length = array.Length;

    for (int i = 0; i <= length - 2; i++) {
        for (int j = length - 1; j >= 1; j--) {

            // 对两个元素进行交换
            if (array[j] < array[j - 1]) {
                T temp = array[j];
                array[j] = array[j - 1];
                array[j - 1] = temp;
            }
        }
    }
}

尽管我们很不情愿,但是问题还是出现了,既然是排序,那么就免不了

要比较大小,大家可以看到在两个元素进行交换时进行了大小的比较,

那么现在请问:book1和book2谁比较大?小张可能说book1大,因为它的

Id是124,而book2的Id是45;而小王可能说book2大,因为它的Title是

以“C”开头的,而book1的Title是以“.”开头的(字符排序时“.”在

“C”的前面)。但是程序就无法判断了,它根本不知道要按照小张的标

准进行比较还是按照小王的标准比较。这时候我们就需要定义一个规则

进行比较。

在.Net中,实现比较的基本方法是实现IComparable接口,它有泛型版本

和非泛型两个版本,因为我们现在正在讲解泛型,为了避免“死锁”,

所以我们采用它的非泛型版本。它的定义如下:

public interface IComparable {
    int CompareTo(object obj);
}

假如我们的Book类型已经实现了这个接口,那么当向下面这样调用时:

book1.CompareTo(book2);

如果book1比book2小,返回一个小于0的整数;如果book1与book2相等

,返回0;如果book1比book2大,返回一个大于0的整数。

接下来就让我们的Book类来实现IComparable接口,此时我们又面对排序

标准的问题,说通俗点,就是用小张的标准还是小王的标准,这里就让

我们采用小张的标准,以Id为标准对Book进行排序,修改Book类,让它

实现IComparable接口:

public class Book :IComparable {
    // CODE:上面的实现略

    public int CompareTo(object obj) {
        Book book2 = (Book)obj;
        return this.Id.CompareTo(book2.Id);
    }
}

为了节约篇幅,我省略了Book类上面的实现。还要注意的是我们并没有

在CompareTo()方法中去比较当前的Book实例的Id与传递进来的Book实

例的Id,而是将对它们的比较委托给了int类型,因为int类型也实现了

IComparable接口。顺便一提,大家有没有发现上面的代码存在一个问题

?因为这个CompareTo ()方法是一个很“通用”的方法,为了保证所有

的类型都能使用这个接口,所以它的参数接受了一个Object类型的参数

。因此,为了获得Book类型,我们需要在方法中进行一个向下的强制转

换。如果你熟悉面向对象编程,那么你应该想到这里违反了Liskov替换

原则,关于这个原则我这里无法进行专门的讲述,只能提一下:这个原

则要求方法内部不应该对方法所接受的参数进行向下的强制转换。为什

么呢?我们定义继承体系的目的就是为了代码通用,让基类实现通用的

职责,而让子类实现其本身的职责,当你定义了一个接受基类的方法时

,设计本身是优良的,但是当你在方法内部进行强制转换时,就破坏了

这个继承体系,因为尽管方法的签名是面向接口编程,方法的内部还是

面向实现编程。

NOTE:什么是“向下的强制转换(downcast)”?因为Object是所有类型

的基类,Book类继承自Object类,在这个金字塔状的继承体系中,

Object位于上层,Book位于下层,所以叫“向下的强制转换”。

好了,我们现在回到正题,既然我们现在已经让Book类实现了

IComparable接口,那么我们的泛型类应该可以工作了吧?不行的,因为

我们要记得:泛型类是一个模板类,它对于在执行时传递的类型参数是

一无所知的,也不会做任何猜测,我们知道Book类现在实现了

IComparable,对它进行比较很容易,但是我们的SortHelper<T>泛型类

并不知道,怎么办呢?我们需要告诉SortHelper<T>类(准确说是告诉编

译器),它所接受的T类型参数必须能够进行比较,换言之,就是实现

IComparable接口,这便是本小节的主题:泛型约束。

为了要求类型参数T必须实现IComparable接口,我们像下面这样重新定

义SortHelper<T>:

public class SortHelper<T> where T:IComparable {
    // CODE:实现略
}

上面的定义说明了类型参数T必须实现IComaprable接口,否则将无法通

过编译,从而保证了方法体可以正确地运行。因为现在T已经实现了

IComparable,而数组array中的成员是T的实例,所以当你在array[i]

后面点击小数点“.”时,VS200智能提示将会给出IComparable的成员,

也就是CompareTo()方法。我们修改BubbleSort()类,让它使用

CompareTo()方法来进行比较:

public class SortHelper<T> where T:IComparable
{
    public void BubbleSort(T[] array) {
        int length = array.Length;

        for (int i = 0; i <= length - 2; i++) {
            for (int j = length - 1; j >= 1; j--) {
                
                // 对两个元素进行交换
                if (array[j].CompareTo(array[j - 1]) < 0 ) {
                    T temp = array[j];
                    array[j] = array[j - 1];
                    array[j - 1] = temp;
                }
            }
        }
    }
}

此时我们再次运行上面定义的代码,会看到下面的输出:

Id:45
Title:.Net之美

Id:124
Title:C# 3.0揭秘

除了可以约束类型参数T实现某个接口以外,还可以约束T是一个结构、T

是一个类、T拥有构造函数、T继承自某个基类等,但我觉得将这些每一

种用法都向你罗列一遍无异于浪费你的时间。所以我不在这里继续讨论

了,它们的概念是完全一样的,只是声明的语法有些差异罢了,而这点

差异,相信你可以很轻松地通过查看MSDN解决。
1.1.3 泛型方法

我们再来考虑这样一个问题:假如我们有一个很复杂的类,它执行多种

基于某一领域的科学运算,我们管这个类叫做SuperCalculator,它的定

义如下:

public class SuperCalculator {
    public int SuperAdd(int x, int y) {
        return 0;
    }

    public int SuperMinus(int x, int y) {
        return 0;
    }

    public string SuperSearch(string key) {
        return null;
    }

    public void SuperSort(int[] array) {
    }
}

由于这个类对算法的要求非常高,.Net框架内置的快速排序算法不能满

足要求,所以我们考虑自己实现一个自己的排序算法,注意到

SuperSearch()和SuperSort()方法接受的参数类型不同,所以我们最好

定义一个泛型来解决,我们将这个算法叫做SpeedSort(),既然这个算法

如此之高效,我们不如把它定义为public的,以便其他类型可以使用,

那么按照前面两节学习的知识,代码可能类似于下面这样:

public class SuperCalculator<T> where T:IComparable {
    // CODE:略

    public void SpeedSort(T[] array) {      
        // CODE:实现略
    }
}

这里穿插讲述一个关于类型设计的问题:确切的说,将SpeedSort()方法

放在SuperCaculator中是不合适的?为什么呢?因为它们的职责混淆了

,SuperCaculator的意思是“超级计算器”,那么它所包含的公开方法

都应该是与计算相关的,而SpeedSort()出现在这里显得不伦不类,当我

们发现一个方法的名称与类的名称关系不大时,就应该考虑将这个方法

抽象出去,把它放置到一个新的类中,哪怕这个类只有它一个方法。

这里只是一个演示,我们知道存在这个问题就可以了。好了,我们回到

正题,尽管现在SuperCalculator类确实可以完成我们需要的工作,但是

它的使用却变得复杂了,为什么呢?因为SpeedSort()方法污染了它,仅

仅为了能够使用SpeedSort()这一个方法,我们却不得不将类型参数T加

到SuperCalculator类上,使得即使不调用SpeedSort()方法时,创建

Calculator实例时也得接受一个类型参数。

为了解决这个问题,我们自然而然地会想到:有没有办法把类型参数T加

到方法上,而非整个类上,也就是降低T作用的范围。答案是可以的,这

便是本小节的主题:泛型方法。类似地,我们只要修改一下SpeedSort()

方法的签名就可以了,让它接受一个类型参数,此时SuperCalculator的

定义如下:

public class SuperCalculator{
    // CODE:其他实现略

    public void SpeedSort<T>(T[] array) where T :

IComparable {
        // CODE:实现略
    }
}

接下来我们编写一段代码来对它进行一个测试:

Book[] bookArray = new Book[2];

Book book1 = new Book(124, "C# 3.0揭秘");
Book book2 = new Book(45, ".Net之美");

SuperCalculator calculator = new SuperCalculator();
calculator.SpeedSort<Book>(bookArray);

因为SpeedSort()方法并没有实现,所以这段代码没有任何输出,如果你

想看到输出,可以简单地把上面冒泡排序的代码贴进去,这里我就不再

演示了。这里我想说的是一个有趣的编译器能力,它可以推断出你传递

的数组类型以及它是否满足了泛型约束,所以,上面的SpeedSort()方法

也可以像下面这样调用:

calculator.SpeedSort(bookArray);

这样尽管它是一个泛型方法,但是在使用上与普通方法已经没有了任何

区别。

理解泛型

标签:

原文地址:http://www.cnblogs.com/1539136475-/p/4598221.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!