您现在的位置是:亿华云 > 系统运维
程序员你如何检查参数的合法性?
亿华云2025-10-09 03:35:51【系统运维】3人已围观
简介作为程序员的你,代码中最多的就是各种方法了,你是如何对参数进行校验的呢?背景大部分的方法和构造函数对传入的参数值有一些限制,比如:常见的索引值必须是非负数,对象引用不能为空。你应该使用清晰的文档来标注
作为程序员的程序参数你,代码中最多的员何就是各种方法了,你是检查如何对参数进行校验的呢?
背景
大部分的方法和构造函数对传入的参数值有一些限制,比如:常见的法性索引值必须是非负数,对象引用不能为空。程序参数
你应该使用清晰的员何文档来标注所有的这些限制,然后在方法体开始的检查地方强制他们检查。
应该在错误发生的法性时候尽快的检查出来,这是程序参数基本原则。
如果你不这么做,员何当错误发生的检查时候,错误将不会被检测出来,法性这让定位错误的程序参数源头变得更困难。
如果一个非法参数传递到一个方法中,员何在方法执行前进行了参数检查。检查它将会快速失败,并给出清晰的免费信息发布网异常信息。
如果方法没有检查参数,下面这些事情会发生。
程度说明
糟糕方法会在执行过程中失败然后抛出一个不明确的异常; 更糟糕方法会正常返回,但是悄悄的计算了一个错误的值。最糟糕方法正常返回,但是一些对象处在一个不正确的状态,未来一个不确定的时间点在某些无关联的点会造成一个错误。
一句话总结:参数不校验会导致原子性失败。
推荐做法
对公共和保护方法,使用java文档的@throws标签来标注参数值不合法将抛出的异常。
常见的参数校验的异常类型如下:
只要你已经已经在文档中标注了方法参数的限制和违反限制会抛出的异常,限制将是一个简单的事情,下面是一个典型的例子。
/** *@param m 必须是亿华云计算正整数 *@throws ArithmeticException 如果m<=0 **/ public BigInteger mod(BigInteger m){ if(m<=0){ throw new ArithmeticException("modulus <=0: "+ m); } //todo 其它代码 }注意:
文档注释并没有说, 如果m是空,mod将抛出NullPointException, 尽管这个方法确实会这样。调用m.signum()的时候这个异常被标注在类级别BigInteger的文档注释上,类级别的注释适用于所有的公共方法的参数,这是一个避免在每个方法单独的文档化标注NullPointException这种混乱的好方法。也许可以结合@Nullable或者类似的注解来指明特殊参数可以为空,但是这个实践并不是标准的,并且有很多注解可以用来达到这个目的。
Objects实用类
Objects.requireNonNull方法,在Java7中添加的,非常的灵活和方便,所以没有理由手动的执行空指针检查。 你也可以指定异常的详细信息,云南idc服务商这个方法返回自己的输入,所以你可以在使用该值的时候执行一个空指针检查。
//一行代码使用java的空指针检查 this.strategy = Objects.requireNonNull(strategy,"strategy")如果你可以忽略返回值,你也可以根据你的需要使用Objects.requireNonNull作为独立的空指针检查。在Java9中,一个范围检查的方法被添加到了java.util.Objects中,包含了3个方法:
这3个方法没有空指针检查方法灵活,它无法让你指定自己的异常详细信息,它被设计用在List和Array的索引检查上。 它也无法处理闭区间,但是只要你需要,这就是一个小便利。
Java断言
对一个不开放的方法,你作为包的作者,控制着方法的调用状况,你必须保证只有合法的参数值传递进去了。所以,对非公开的方法,你可以使用断言来进行参数检查,如下所示:
//私有帮助排序函数 private static void sort(long a[] , int offset, int length){ assert a != null ; //更多代码 }本质上来讲,断言申明条件一定是true , 忽略客户端如何使用对应的包。跟一般的合法性检查不同,断言失败的时候抛出AssertError;跟一般的合法性检查不同,除非你启用他们否则断言对你没有任何影响和消耗。在java命令行启用指令:
-ea 或者 -enableassertions更多断言的信息,查看java手册的Asserts;
检查参数的合法性非常重要,即使你的方法中没有用到,但是存储起来了,后面会用到。
举个例子: 静态工厂方法: 输入一个 int数组 ,返回一个array的 list视图, 如果客户端传入 null, 这个方法会抛出NPE, 因为方法会有一个直接检查,调用了Objects.requireNonNull。如果忽略检查,方法会返回一个引用新创建的List的实例;
而客户端尝试使用的时候回抛出NPE; 这个时候,原始的List实例很难决定,很大可能会复杂到变成一个调试任务。
构造函数代表了一个特殊例子的原则: 你应该检查即将存储稍后会用到的参数的合法性。
检查构造函数参数的合法性非常重要,它可以防止构造一个违反类的不变性的对象。
异常情况
在执行方法计算之前,你应该检查方法参数 。 这个规则也有异常情况。
一个重要的异常情况是:合法性检查代价非常高并且重要, 并且检查是在执行计算的过程中执行的。举个例子:有一个方法对一个对象list排序,比如 Collectios.sort(list),所有的list中的对象必须是可互相比较的。在处理list比较的时候,每个对象将会跟其它的对象进行比较,
如果对象不能互相比较,其中一个或多个比较会抛出ClassCastException,这是排序方法应该做的。
所以:这里有一个小店,在开始的时候检查列表中的元素应该是可以互相比较的,注意:修改合法性检查会丧失原子失败。
偶尔,一个计算执行了一个需要的合法性检查,但是当执行检查失败的时候,抛出了一个错误的异常。换句话说,计算常常会抛出参数合法性检查的异常,并不会匹配方法在文档中申明的异常。这种场景下,你应该使用异常翻译成语。 转换自然异常为正确的异常。
这个原则并不是说武断的限制参数是一件好事,而是说:你应该设计通用实际的方法。假设你的方法接受所有的参数组合而可以做一些合理事情,你的参数限制越少越好,然而,一些限制本质上在抽象类中已经被实现了。
小结
如果看完之后你只能记住一句话:每次你写一个方法或者一个构造函数,你应该思考参数的限制是否存在,你应该把限制写在文档中,并在方法体的开始部分确保进行了检查。
养成这个习惯很重要,适当的工作会在第一次合法性检查失败的时候回馈你。

很赞哦!(27564)
相关文章
- 3、商标域名一经注册,就可以作为域名裁决过程中的主要信息之一。这可以大大增加公司被抢注的相关域名胜诉的机会。
- 使用CSS Flexbox 构建可靠实用的网站 Header
- 架构师的成长之路,第一步该怎么迈?我给大家准备好了
- 效率翻倍!给开发人员的几点建议
- 域名不仅仅是一个简单的网站。对于有长远眼光的公司来说,在运营网站之前确定一个优秀的域名对有长远眼光的公司来说是非常重要的。这对今后的市场营销、产品营销和企业品牌建设都具有十分重要的意义。优秀的域名是企业在市场竞争中获得持久优势的利器。
- GitHub率先消灭了cookies:与烦人的用户条款说再见
- K8s宣布弃用Docker,千万别慌!
- 2020征文-开发板鸿蒙Hi3861之俄罗斯方块小游戏(附源码)
- 为了避免将来给我们的个人站长带来的麻烦,在选择域名后缀时,我们的站长最好省略不稳定的后缀域名,比如n,因为我们不知道策略什么时候会改变,更不用说我们将来是否还能控制这个域名了。因此,如果站长不是企业,或者有选择的话,如果不能选择域名的cn类,最好不要选择它。
- Python有什么用?2020年学习Python的10个理由
热门文章
站长推荐
用户邮箱的静态密码可能已被钓鱼和同一密码泄露。在没有收到安全警报的情况下,用户在适当的时间内不能更改密码。在此期间,攻击者可以随意输入帐户。启用辅助身份验证后,如果攻击者无法获取移动电话动态密码,他将无法进行身份验证。这样,除非用户的电子邮件密码和手机同时被盗,否则攻击者很难破解用户的邮箱。
2020征文-开发板鸿蒙Hi3861之俄罗斯方块小游戏(附源码)
TIOBE12月榜单:Java重回第二,Python有望四连冠年度语言
高性能解决线程饥饿的利器 StampedLock
如果你的潜在终端必须是这个米(域名),那么潜在终端并不多,也没有硬通货,那么你的域名应该在终端有兴趣购买时出售。否则,你可能得自己留着吃。
划入 .NET 6 版本目标,微软鼓励开发人员信任第三方库
9个好用的JavaScript小技巧
Java语言未来是否还有发展前景