规范约束条件

规范约束条件我们在开发时往往会对泛型指定约束条件,只有类型参数符合条件的才允许用在这个泛型上面。但是有时我们会定义过多或过少的约束条件,过多的约束条件会导致其他开发人员在使用你所编写的方法或类时做很多的工作以满足这些约束,过少的约束又会导致程序在运行的时候必须做很多的检查,并执行更多的强制类型转化操作,有时我们还需要使用反射生成运行期错误,来防止用户误用这个类。要解决这些问题,我们就必须把确实需要的约束写出来…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

我们在开发时往往会对泛型指定约束条件,只有类型参数符合条件的才允许用在这个泛型上面。但是有时我们会定义过多或过少的约束条件,过多的约束条件会导致其他开发人员在使用你所编写的方法或类时做很多的工作以满足这些约束,过少的约束又会导致程序在运行的时候必须做很多的检查,并执行更多的强制类型转化操作,有时我们还需要使用反射生成运行期错误,来防止用户误用这个类。要解决这些问题,我们就必须把确实需要的约束写出来,这句话说起来简单,其实做起来不太容易。下面我就来讲解一下如何正确的编写一个规范的约束。

零、简述

何为约束?所谓约束就是使得编译器能够知道 类型参数 除了具备 System.Object 所定义的公共接口外还需要满足的条件。在创建泛型类型时编译器必须为这个泛型类型定义有效的 IL 码,即使它不知道其中的类型参数会在什么时候替换成什么类型,也会设法创建出有效的程序集。如果我们不给它指明类型参数,那么它就会默认设置类型参数是 System.Object 类型。我们通过约束来表达对泛型类型的类型参数的约束要求会营销编译器和使用这个类的开发人员。编译器看到我们指定的约束后就会明白除了除了具备 System.Object 所定义的公共接口外还需要满足什么条件。对于编译器来说它获得了两个帮助:

  1. 可以令编译器在创建这个泛型类型的时候获得更多的信息;
  2. 编译器能够保证使用这个泛型的开发人员所提供的参数类型一定满足我们所指定的条件。

一、如何规范约束条件

讲解之前我们先来看一个例子,这个例子判断了输入的两个值是否相等。

public bool DemoEqual<T>(T t1, T t2)
{ 
   
    if(t1==null)
    { 
   
        return t2==null;
    }
    if(t1 is IComparable<T>)
    { 
   
        IComparable<T> val1 = t1 as IComparable<T>;
        if(t2 as IComparable<T>)
        { 
   
            return val1.CompareTo(t2)==0;
        }
        else
        { 
   
            throw new ArgumentException($"{ 
     nameof(t2)} 没有实现 IComparable<T>");
        }
    }
    else
    { 
   
        throw new ArgumentException($"{ 
     nameof(t1)} 没有实现 IComparable<T>")
    }
}

这段代码中执行了大量的强类型转换,在转换之前还判断时传入的参数是否实现了 IComparable 接口。这段代码如果使用了泛型约束就会很简单:

public bool DemoEqual<T>(T t1, T t2) 
    where T : IComparable<T>
        => t1.CompareTo(t2)==0;

这段代码大大简化了前面的那段代码,并且把程序运行期可能出现的错误提前到了编译期,编译器提前阻止了不符合要求的用法。到这里你是不是以为上述代码就是很好的解决方案呢?其实严格来说上述代码矫枉过正了,为什么这么说呢?因为 IComparable 接口很常见,大部分开发人员在设计类型的时候都会事先这个接口,因此我们将上述代码修改一些,我们不使用 CompareTo 来对比两个值是否相等,我们这次使用 Equals 来对比:

public bool DemoEqual<T>(T t1, T t2) 
    => t1.Equals(t2);

上述代码有一点需要注意,如果 DemoEqual 是定义在泛型类里,并且泛型类也规定了 IComparable 约束,那么他调用的 Equals 是 IComparable.Equals ,反之调用的就是 System.Object.Equals 。这两个 Equals 在性能上没什么大的差别,前者的执行效率只比后者高了那么一丢丢,因为它只是不用在运行时检查程序有没有重写 System.Object.Equals ,以及泛型参数类型为值类型时它也不用执行装箱和拆箱操作。但是对于把性能看的特别重的开发人员来说,前者是最优的方案。

Tip:如果有较好的方法,我还是建议大家使用较好的方法,比如前面我们所说的 IComparable.Equals 。

我们在编写泛型类的时候,最好在内部编写相互重载的多个方法,这样就可以针对不同的情况调用不同的方法,并且其他开发人员调用起来也不会有过于严谨的约束。有时候我们定义的约束过于严谨,会导致泛型类的适用范围很狭窄,遇到这种情况时我们就应该考虑我们自己在泛型类种编写代码来判断传入的类型是否继承自某个类或者实现了某个接口。在泛型约束中有三种约束我们必须谨慎使用,它们就是 new 、 struct 以及 class 约束,因为它们会限定对象的构建方式,除非你要求对象的默认值必须是 0 、null 或者必须能以 new() 的形式创建,那么我们才可以使用这三种约束。

二、总结

约束是为了向调用方提出要求,但是如果约束太多调用方就需要做更多的工作来满足这些约束,因此在创建约束时应该权衡利弊,将多余的约束去掉只保留需要的约束。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/182362.html原文链接:https://javaforall.net

(0)
上一篇 2022年10月13日 下午6:00
下一篇 2022年10月13日 下午6:00


相关推荐

  • 文本分类算法综述

    文本分类算法综述文本分类大致有两种方法:一种是基于训练集的文本分类方法;另一种是基于分类词表的文本分类方法。两种方法出自不同角度的研究者,训练集法更多的来自计算机或人工智能研究领域,而分类表法则更多地来自突出情报领域。本文主要介绍前一种。基于训练集的文本分类是一种典型的有教师的机器学习问题,一般分为训练和分类两个阶段,具体过程如下:训练阶段:1)             定义类别集合 ,这些类别可是是层次式的,…

    2022年6月9日
    37
  • 字节跳动社招面试经验_字节hr面试后多久会通知你

    字节跳动社招面试经验_字节hr面试后多久会通知你虽然已经临近年末,但是还是萌生要看新机会的想法,主要的原因是觉得在目前的岗位上技术增长遇到的瓶颈,因此想去做一些更有挑战的工作。因为仍然准备继续在深圳工作,因此选定了三家公司,腾讯、字节跳动和sho

    2022年8月3日
    10
  • 第一部分:欢迎来到 n8n → 入门指南:n8n 的用户界面 → 工作流编辑器 → 输出面板

    第一部分:欢迎来到 n8n → 入门指南:n8n 的用户界面 → 工作流编辑器 → 输出面板

    2026年3月15日
    2
  • 常见sql注入语句和xss注入语句

    常见sql注入语句和xss注入语句1and1 1 或者 1 and 1 1 判断注入的类型 1orderby1 2 3 4 数字类型为例 判断当前表存在多少个属性字段 1 unionselect1 table namefrominfo schema tableswheret schema database 通过占位获取的数据库名去 MySQL 的元数据表 information schema 中查表名 1 unionselect1 c

    2026年3月16日
    1
  • ArcGIS for Android 100.3.0(8):绘制点,线,面,圆,添加文本和图片「建议收藏」

    ArcGIS for Android 100.3.0(8):绘制点,线,面,圆,添加文本和图片「建议收藏」空间要素(Geometry)Geometries用以在特定地理位置上通过形状来表达真实世界的对象。图层范围、视图范围、GPS定位都是通过Geometries表达实现进一步的数据编辑、空间分析、地理处理、位置与面积量算都离不开空间要素。案例效果图:布局:&lt;?xmlversion="1.0"encoding="utf-8"?&gt;&lt;RelativeL…

    2022年7月16日
    22
  • [苹果开发者账号]06 转让开发者账号后,开发者年费自动续费问题

    [苹果开发者账号]06 转让开发者账号后,开发者年费自动续费问题由于用户 A 又开通了自动续费 所以到了续费时间 又自动扣了用户 A688 元 但用户 B 对应的苹果开发者账号并没有续费 但用户 A 这时候都已经不是苹果开发者了 结果扣了用户 A 年费 688 元 用户 A 已经不是苹果开发者了 不存在续费开发者账号了 1 由于用户 A 从公司离职了 所以把用户 A 的苹果开发者账号转让给了用户 B 绑定了用户 B 的 AppleID 1 用户 A 是某公司的苹果开发者账号的持有人 绑定了用户 A 的 AppleID 原因账号转让的功能 没有把用户 A 订阅的开发者账号业务转让给用户 B 2 用户 B 变成为苹果开发者

    2026年3月19日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号