zoukankan      html  css  js  c++  java
  • 泛型中的类型约束和类型推断

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

    类型约束

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

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

    总结

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

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

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

  • 相关阅读:
    C连载5-函数与return拾贝
    Android连载15-复习以往内容(一)
    Spring security OAuth2.0认证授权学习第四天(SpringBoot集成)
    Spring security OAuth2.0认证授权学习第三天(认证流程)
    Spring security OAuth2.0认证授权学习第二天(基础概念-RBAC)
    Spring security OAuth2.0认证授权学习第二天(基础概念-授权的数据模型)
    Spring security OAuth2.0认证授权学习第一天(基础概念-认证授权会话)
    JVM学习第三天(JVM的执行子系统)之类加载机制补充
    FastJSON解析JSON的时候保证深堆的顺序
    JVM学习第三天(JVM的执行子系统)之类加载机制
  • 原文地址:https://www.cnblogs.com/wilber2013/p/4292240.html
Copyright © 2011-2022 走看看