泛型中的类型约束和类型推断

前一篇文章介绍了泛型的基本概念。在本文中,我们看一下泛型中两个很重要的特性:类型约束和类型推断。

类型约束

相信你还记得前面一篇文章中的泛型方法,在这个泛型方法中,我们就使用了类型约束。

类型约束(type constraint)进一步控制了可指定的类型实参,当我们创建自己的泛型类型或者泛型方法的时候,类型约束是很有用的。

回到前一篇例子中的泛型方法,这个泛型方法就要求可指定的类型实参必须实现了IComparable接口。

为什么会有这个约束呢?原因很简单,因为我们在泛型方法的实现中直接调用T类型的"CompareTo"方法。所以,我们需要通过一个约束来保证T类型都有"CompareTo"方法,也就是说我们要指定的类型实参T要实现IComparable接口。

public static T GetBiggerOne<T>(T itemOne, T itemTwo) where T : IComparable
{
    if (itemOne.CompareTo(itemTwo) > 0)
    {
        return itemOne;
    }
    return itemTwo;
}

经过上面的解释,大家肯定对约束有了简单的认识。

在类型约束中,有四种约束可供使用,他们的语法都是基本相同的,约束要放到泛型类型或泛型方法的末尾,并由上下文关键字where来引入。同时,约束也可以按照一定的规则组合在一起使用。

下面我们就分别看看可供我们使用的四种类型约束。

引用类型约束

引用类型表示为T : class,用于确保指定的类型实参都是引用类型(任何类,接口,数组或委托,以及已知为引用类型的另一个类型参数)。

如果使用引用类型约束,那么它必须是为类型参数指定的第一个约束

一个简单的示例,例如对于下面的声明:

struct RefSample<T> where T : class { }

有效的封闭类型:

  • RefSample<string>
  • RefSample<IDisposable>

无效的封闭类型:

  • RefSample<int>
  • RefSample<double>

值类型约束

跟引用类型约束形式类似,值类型约束表示为T : struct,用于确保指定的类型实参都是值类型。

同样,如果使用值类型约束,那么它必须是为类型参数指定的第一个约束

例如对于下面的声明:

class ValSample<T> where T : struct { }

有效的封闭类型:

  • ValSample <int>

无效的封闭类型:

  • ValSample <string>

构造函数类型约束

构造函数类型约束表示为T : new(),用于确保所有的类型参数有一个无参数的构造函数,这个构造函数可用于创建类型的实例。这适用于:所有值类型;所有非静态、非抽象、没有显示声明的构造函数的类;显示声明了一个公共无参构造函数的所有非抽象类。

如果使用构造函数类型约束,那么它必须是为类型参数指定的最后一个约束

下面用一个例子进行简单的说明:

public T CreateInstance<T>() where T : new()
{
    return new T();
}

这次例子中是一个泛型方法,约束我们指定的类型实参必须拥有无参数的构造函数,在这种情况下,这个泛型方法就可以返回该类型的一个实例。

所有下面都是有效的调用:

  • CreateInstance<int>()
  • CreateInstance<object>()

注意,在C#中,所有的值类型都有一个默认的无参数构造函数,所以当我们使用一些组合约束的时候,C#编译器就会报出一个错误,因为这样的指定是多余的,所有值类型都隐式提供一个无参公共构造函数。

public T CreateInstance<T>() where T: struct, new()

转换类型约束

转型类型约束允许我们指定另一个类型,类型实参必须可以通过一致性、引用或装箱转换隐式的转换为改类型。

根据上面的描述,可以看到转换类型约束可以有以下一些表示:

  • T : <基类名>,类型参数必须是指定的基类或派生自指定的基类
  • T : <接口名>,类型参数必须是指定的接口或实现指定的接口;可以指定多个接口约束;约束接口也可以是泛型的
  • T : U,用于指定T的类型实参必须是用于指定U的类型实参或者派生自用于指定U的类型实参

下面看几个例子:

class Sample<T> where T: Stream 

有效:Sample<Stream>

无效:Sample<string>

class Sample<T> where T:  IDisposable 

有效:Sample<SqlConnection >

无效:Sample<StringBuilder>

class Sample<T,U> where T: U

有效:Sample<Stream,IDispsable>

无效:LSample<string,IDisposable>

 

组合约束

组合约束就是将前面提到的多种约束集合起来使用。

对于一个类型参数,我们可以使用where关键字进行多个约束;对于不同的类型参数,可以有不同的约束,它们分别由单独的where关键字引入。

在组合约束中,有很多组合情况是无效的,下面看一下例子:

  • class Sample<T> where T: class, struct
    • 没有任何类型即时引用类型又是值类型的,所以这样的组合是无效的
  • class Sample<T> where T: Stream, class
    • 引用类型约束应该为第一个约束,所以这样的组合无效的(同样,如果使用值类型约束,也必须是第一个)
  • class Sample<T> where T: new(), Stream
    • 构造函数约束必须放在最后面,所以这样的组合无效的
  • class Sample<T> where T: IDisposable, Stream
    • 如果存在多个转换类型约束,并且其中一个为类,那么它应该出现在接口的前面
  • class Sample<T> where T: XmlReader, IComparable, IComparable
    • 对于转换类型约束,同一个接口不能出现多次
  • class Sample<T,U> where T: struct where U:class, T
    • 类型形参"T"具有"struct"约束,因此"T"不能用作"U"的约束
  • class Sample<T,U> where T:Stream, U:IDisposable
    • 不同的类型参数可以有不同的约束,它们必须分别由单独的where关键字引入

类型推断

在调用泛型方法的时候,我们都需要通过"<>"来指定类型实参,就会显得代码比较复杂、冗余。其实,根据方法调用时传递的实参类型,可以比较容易的推断出泛型方法的类型参数应该是什么。

所以,编译器就添加了一些"智能",帮我们推断泛型方法的类型参数,这样我们在方法调用的时候就可以不用显示的声明类型实参。

注意,类型推断只适用于泛型方法。

看一个简单的类型推断的例子:

class Program
{
    static void Main(string[] args)
    {
        //Console.WriteLine("The bigger one is {0}", GetBiggerOne<int>(3, 9));
        //Console.WriteLine("The bigger one is {0}", GetBiggerOne<string>("Hello", "World"));
        //让编译器进行类型推断
        Console.WriteLine("The bigger one is {0}", GetBiggerOne(3, 9));
        Console.WriteLine("The bigger one is {0}", GetBiggerOne("Hello", "World"));

        Console.Read();
    }

    public static T GetBiggerOne<T>(T itemOne, T itemTwo) where T : IComparable
    {
        if (itemOne.CompareTo(itemTwo) > 0)
        {
            return itemOne;
        }
        return itemTwo;
    }
}

总结

本文中介绍了泛型中的类型约束和类型推断特性。

在我们使用自定义的泛型类型和泛型方法的时候,如果我们已经发现需要进行一些约束,最好就是直接在声明泛型类型和方法的时候把约束加上。同时应该注意组合约束中的一系列无效的约束组合。

对于类型推断,这个特性只适合泛型方法,可以简化我们调用泛型方法时显示的声明类型实参。

 

posted @ 2015-02-14 22:31  田小计划  阅读(2445)  评论(0编辑  收藏  举报
Fork me on GitHub