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

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

类型约束

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

类型约束(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;
    }
}

总结

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

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

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

时间: 2024-08-26 06:32:23

泛型中的类型约束和类型推断的相关文章

对C#泛型中的new()约束思考

对于new()约束,大家可能有一个误解,以为使用了new约束之后,在创建对象时与非泛型的版本是一致的:<public class Tester<T> where T:new() { public Tester() { t = new T();//等同于非泛型版本的new? 例如 object o = new object();? } private T t; }  事实上,使用new关键字的作用只是让编译器在泛型实例化之处,检查所绑定的泛型参数是否具有无参构造函数: Tester<

类型约束

php类型约束入门详解 众所周知,php是一种弱类型的编程语言.在php程序中,变量的数据类型可以随着其值的不同而自动发生改变,php也不会对变量的数据类型进行强制检查或约束.我们可以参考下面一个简单的代码示例: <?php class Person { } $a = 1; //此时,$a为整数类型型(Integer) var_dump($a); $a = 1.0; //此时,$a为浮点类型(Float) var_dump($a); $a = 'CodePlayer'; //此时,$a为字符串类

java泛型中的上、下界通配符

java泛型中,通配符?表示未知类型,等同于<? extends Object>,<? extends T>是上边界限定通配符,<? super T>是下边界限定通配符. 一.区别 在一个list中,上下界通配符能够存放和读取的对象类型如下图所示: 二.原则 上下界通配符的使用应当遵循PECS原则:Producer Extends,Consumer Super. 限定通配符总是包括自己 上界类型通配符:add方法受限 下界类型通配符:get方法受限 如果你想从一个数据类

Scala 深入浅出实战经典 第48讲:Scala类型约束代码实战及其在Spark中的应用源码解析

王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-64讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 腾讯微云:http://url.cn/TnGbdC 360云盘:http://yunpan.cn/cQ4c2UALDjSKy 访问密码 45e2 技术爱好者尤其是大数据爱好者 可以加DT大数据梦工厂的qq群 DT大数据梦工厂① :462923555 DT大数据梦工厂②:437123764 DT大数据梦工厂③

第81讲:Scala中List的构造和类型约束逆变、协变、下界详解

今天来学习一下scala中List的构造和类型约束等内容. 让我们来看一下代码 package scala.learn /** * @author zhang */abstract class Big_Data class Hadoop extends Big_Dataclass Spark extends Big_Data object List_Constructor {  def main(args:Array[String]){    val hadoop = new Hadoop ::

Scala 深入浅出实战经典 第81讲:Scala中List的构造是的类型约束逆变、协变、下界详解

王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-97讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 腾讯微云:http://url.cn/TnGbdC 360云盘:http://yunpan.cn/cQ4c2UALDjSKy 访问密码 45e2土豆:http://www.tudou.com/programs/view/ceac2IoB-ik/优酷:http://v.youku.com/v_show/id_

Scala中List的构造是的类型约束逆变、协变、下界详解

学习了Scala中List的构造是的类型约束逆变.协变.下界详解,列表中用::加入父类的对象,列表会协变为父类,例子如下: Def :: [B>:A](x:B):List(B)= New scala:collection.imutable.::(x,this) 王家林亲授<DT大数据梦工厂>大数据实战视频“Scala深入浅出实战经典”视频.音频和PPT下载!第81讲:Scala中List的构造是的类型约束逆变.协变.下界详解腾讯微云:http://url.cn/UNeLA2百度云盘:ht

JAVA泛型中的类型擦除及为什么不支持泛型数组

一,数组的协变性(covariant array type)及集合的非协变性 设有Circle类和Square类继承自Shape类. 关于数组的协变性,看代码: public static double totalArea(Shape[] arr){ double total = 0; for (Shape shape : arr) { if(shape != null) total += shape.area(); } return total; } 如果给 totalArray(Shape[

泛型中的类型擦除

通过反射理解泛型的本质(类型擦除) Java中的泛型是通过类型擦除来实现的.所谓类型擦除,是指通过类型参数合并,将泛型类型实例关联到同一份字节码上.编译器只为泛型类型生成一份字节码,并将其实例关联到这份字节码上.类型擦除的关键在于从泛型类型中清除类型参数的相关信息,并且再必要的时候添加类型检查和类型转换的方法. 下面通过两个例子来证明在编译时确实发生了类型擦除. 例1分别创建实际类型为String和Integer的ArrayList对象,通过getClass()方法获取两个实例的类,最后判断这个