本篇文章给大家分享的是有关如何在Java代码中去掉烦人的”!=null”,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。
问题
为了避免空指针调用,我们经常会看到这样的语句
<代码>如果(someobject !=null) {
,,someobject.doCalc ();
}
代码>
最终,项目中会存在大量判空代码,多么丑陋繁冗!如何避免这种情况?我们是否滥用了判空呢?
回答
这是初,中级程序猿经常会遇到的问题。他们总喜欢在方法中返回零,因此,在调用这些方法时,也不得不去判空。另外,也许受此习惯影响,他们总潜意识地认为,所有的返回都是不可信任的,为了保护自己程序,就加了大量的判空。
吐槽完毕,回到这个题目本身,进行判空前,请区分以下两种情况:
- <李>
零是一个有效有意义的返回值(空是一个有效的响应在合同条款;李和)
> <李>零是无效有误的(这不是一个有效的回应。)
李>你可能还不明白这两句话的意思,不急,继续往下看,接下来将详细讨论这两种情况
先说第2种情况
零就是一个不合理的参数,就应该明确地中断程序,往外抛错误。这种情况常见于api的方法,例如你开发了一个接口,id是一个必选的参数,如果调用方没传这个参数给你,当然不行。你要感知到这个情况,告诉调用方“嘿,哥们,你传个空给我做甚“。
相对于判空语句,更好的检查方式有两个
- <李>
断言语句,你可以把错误原因放到维护的参数中,这样不仅能保护你的程序不往下走,而且还能把错误原因返回给调用方,岂不是一举两得。(原文介绍了断言的使用,这里省略)
李> <李>也可以直接抛出空指针异常上。面说了,此时零是个不合理的参数,有问题就是有问题,就应该大大方方往外抛。
李>
第1种情况会更复杂一些。
这种情况下,零是个“看上去“合理的值,例如,我查询数据库,某个查询条件下,就是没有对应值,此零算时是表达了“空”的概念。
这里给一些实践建议:
假如方法的返回类型是集合,当返回结果是空时,你可以返回一个空的集合(空列表),而不要返回零。这样调用侧就能大胆地处理这个返回,例如调用侧拿到返回后,可以直接打印list.size(),又无需担心空指针问题。(什么?想调用这个方法时,不记得之前实现该方法有没按照这个原则?所以说,代码习惯很重要。如果你养成习惯,都是这样写代码(返回空集合而不返回零),你调用自己写的方法时,就能大胆地忽略判空)
返回类型不是收藏,又怎么办呢?
那就返回一个空对象(而非零对象),下面举个“栗子”,假设有如下代码
<代码>公共接口操作{
,空白doSomething ( );}
公共接口的解析器{
,行动 findAction (String userInput);}
代码>
其中,解析有一个接口findAction,这个接口会依据用户的输入,找到并执行对应的动作。假如用户输入不对,可能就找不到对应的动作(行动),因此findAction就会返回零,接下行动来调用doSomething方法时,就会出现空指针。
解决这个问题的一个方式,就是使用null对象模式(空对象模式)
我们来改造一下
类定义如下,这样定义findAction方法后,确保无论用户输入什么,都不会返回零对象:
<代码>公共类MyParser实现解析器{
,私有静态动作DO_NOTHING=新的行动(){
,,公共空doSomething () {/*不*/}
,},
,公共行动 findAction(字符串userInput) {
,,//?br/>,我们停下来,如果(/* # 39;t找到任何操作*/){
,,,返回DO_NOTHING;
,,}
}
}
代码>
对比下面两份调用实例
1。冗余:每获取一个对象,就判一次空
<代码>=ParserFactory.getParser解析器解析器();如果(解析器==
null) {
,//现在是什么?
,//这是一个例子 在哪里 空不# 39;t(或者应该# 39;t是一个有效的回应
}
行动Action=parser.findAction (someInput);如果(Action==
null) {
,//什么都不做} 其他
{
, action.doSomething ();
}
代码>
2。精简
<代码> ParserFactory.getParser () .findAction (someInput) .doSomething ();
代码>
因为无论什么情况,都不会返回空对象,因此通过findAction拿到动作后,可以放心地调用行动的方法。扩展一下:Java:如何更优雅的处理空值吗?