Java中如何解决null值引起的Bug

发布时间:2021-11-01 14:23:02 作者:iii
来源:亿速云 阅读:141

这篇文章主要讲解了“Java中如何解决null值引起的Bug”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java中如何解决null值引起的Bug”吧!

业务中的空值

场景

存在一个 UserSearchService用来提供用户查询的功能: 

public interface UserSearchService{       List<User> listUser();       User get(Integer id);     }

问题现场

对于面向对象语言来讲,抽象层级特别的重要。尤其是对接口的抽象,它在设计和开发中占很大的比重,我们在开发时希望尽量面向接口编程。

对于以上描述的接口方法来看,大概可以推断出可能它包含了以下两个含义:

在所有的开发中,XP推崇的TDD模式可以很好的引导我们对接口的定义,所以我们将TDD作为开发代码的”推动者”。

对于以上的接口,当我们使用TDD进行测试用例先行时,发现了潜在的问题:

深入listUser研究

我们先来讨论 

listUser()

这个接口,经常看到如下实现:

public List<User> listUser(){        List<User> userList = userListRepostity.selectByExample(new UserExample());        if(CollectionUtils.isEmpty(userList)){//spring util工具类          return null;        }        return userList;    }

这段代码返回是null,从我多年的开发经验来讲,对于集合这样返回值,最好不要返回null,因为如果返回了null,会给调用者带来很多麻烦。你将会把这种调用风险交给调用者来控制。

如果调用者是一个谨慎的人,他会进行是否为null的条件判断。如果他并非谨慎,或者他是一个面向接口编程的狂热分子(当然,面向接口编程是正确的方向),他会按照自己的理解去调用接口,而不进行是否为null的条件判断,如果这样的话,是非常危险的,它很有可能出现空指针异常!

基于此,我们将它进行优化:

public List<User> listUser(){        List<User> userList = userListRepostity.selectByExample(new UserExample());        if(CollectionUtils.isEmpty(userList)){          return Lists.newArrayList(); //guava类库提供的方式        }        return userList;    }

对于接口( ListlistUser()),它一定会返回List,即使没有数据,它仍然会返回List(集合中没有任何元素);

通过以上的修改,我们成功的避免了有可能发生的空指针异常,这样的写法更安全!

深入研究get方法

对于接口

User get(Integer id)

你能看到的现象是,我给出id,它一定会给我返回User.但事实真的很有可能不是这样的。

我看到过的实现:

public User get(Integer id){      return userRepository.selectByPrimaryKey(id);//从数据库中通过id直接获取实体对象    }

相信很多人也都会这样写。

通过代码的时候得知它的返回值很有可能是null! 但我们通过的接口是分辨不出来的!

这个是个非常危险的事情。尤其对于调用者来说!

我给出的建议是,需要在接口明明时补充文档,比如对于异常的说明,使用注解@exception: 

public interface UserSearchService{        /**         * 根据用户id获取用户信息         * @param id 用户id         * @return 用户实体         * @exception UserNotFoundException         */        User get(Integer id);      }

我们把接口定义加上了说明之后,调用者会看到,如果调用此接口,很有可能抛出“UserNotFoundException(找不到用户)”这样的异常。

这种方式可以在调用者调用接口的时候看到接口的定义,但是,这种方式是”弱提示”的!

如果调用者忽略了注释,有可能就对业务系统产生了风险,这个风险有可能导致一个亿!

除了以上这种”弱提示”的方式,还有一种方式是,返回值是有可能为空的。那要怎么办呢?

我认为我们需要增加一个接口,用来描述这种场景.

引入jdk8的Optional,或者使用guava的Optional.看如下定义: 

public interface UserSearchService{       /**        * 根据用户id获取用户信息        * @param id 用户id        * @return 用户实体,此实体有可能是缺省值        */       Optional<User> getOptional(Integer id);     }

Optional有两个含义: 存在 or 缺省。

那么通过阅读接口getOptional(),我们可以很快的了解返回值的意图,这个其实是我们想看到的,它去除了二义性。

它的实现可以写成:

public Optional<User> getOptional(Integer id){    return Optional.ofNullable(userRepository.selectByPrimaryKey(id));  }

深入入参

通过上述的所有接口的描述,你能确定入参id一定是必传的吗?我觉得答案应该是:不能确定。除非接口的文档注释上加以说明。

那如何约束入参呢?

推荐两种方式:

1.强制约束,我们可以通过jsr 303进行严格的约束声明: 

public interface UserSearchService{       /**        * 根据用户id获取用户信息        * @param id 用户id        * @return 用户实体        * @exception UserNotFoundException        */       User get(@NotNull Integer id);       /**        * 根据用户id获取用户信息        * @param id 用户id        * @return 用户实体,此实体有可能是缺省值        */       Optional<User> getOptional(@NotNull Integer id);     }

当然,这样写,要配合AOP的操作进行验证,但让spring已经提供了很好的集成方案,在此就不在赘述了。

2.文档性约束

在很多时候,我们会遇到遗留代码,对于遗留代码,整体性改造的可能性很小。

我们更希望通过阅读接口的实现,来进行接口的说明。

jsr 305规范,给了我们一个描述接口入参的一个方式(需要引入库 com.google.code.findbugs:jsr305):

可以使用注解: @Nullable @Nonnull @CheckForNull 进行接口说明。比如:

public interface UserSearchService{     /**      * 根据用户id获取用户信息      * @param id 用户id      * @return 用户实体      * @exception UserNotFoundException      */     @CheckForNull     User get(@NonNull Integer id);     /**      * 根据用户id获取用户信息      * @param id 用户id      * @return 用户实体,此实体有可能是缺省值      */     Optional<User> getOptional(@NonNull Integer id);   }

小结

通过 空集合返回值,Optional,jsr 303,jsr 305这几种方式,可以让我们的代码可读性更强,出错率更低!

空对象模式

场景

来看一个DTO转化的场景,对象: 

@Data     static class PersonDTO{       private String dtoName;       private String dtoAge;     }     @Data     static class Person{      private String name;       private String age;    }

需求是将Person对象转化成PersonDTO,然后进行返回。

当然对于实际操作来讲,返回如果Person为空,将返回null,但是PersonDTO是不能返回null的(尤其Rest接口返回的这种DTO)。

在这里,我们只关注转化操作,看如下代码:

@Test   public void shouldConvertDTO(){     PersonDTO personDTO = new PersonDTO();     Person person = new Person();     if(!Objects.isNull(person)){       personDTO.setDtoAge(person.getAge());       personDTO.setDtoName(person.getName());     }else{       personDTO.setDtoAge("");       personDTO.setDtoName("");     }   }

优化修改

这样的数据转化,可读性非常差,每个字段的判断,如果是空就设置为空字符串(“”)

换一种思维方式进行思考,我们是拿到Person这个类的数据,然后进行赋值操作(setXXX),其实是不关系Person的具体实现是谁的。

那我们可以创建一个Person子类:

static class NullPerson extends Person{      @Override      public String getAge() {        return "";      }      @Override      public String getName() {        return "";      }    }

它作为Person的一种特例而存在,如果当Person为空的时候,则返回一些 get*的默认行为.

所以代码可以修改为: 

@Test      public void shouldConvertDTO(){        PersonDTO personDTO = new PersonDTO();        Person person = getPerson();        personDTO.setDtoAge(person.getAge());        personDTO.setDtoName(person.getName());      }      private Person getPerson(){        return new NullPerson();   //如果Person是null ,则返回空对象     }

其中 getPerson()方法,可以用来根据业务逻辑获取Person有可能的对象(对当前例子来讲,如果Person不存在,返回Person的的特例NUllPerson),如果修改成这样,代码的可读性就会变的很强了。

使用Optional可以进行优化

空对象模式,它的弊端在于需要创建一个特例对象,但是如果特例的情况比较多,我们是不是需要创建多个特例对象呢,虽然我们也使用了面向对象的多态特性,但是,业务的复杂性如果真的让我们创建多个特例对象,我们还是要再三考虑一下这种模式,它可能会带来代码的复杂性。

对于上述代码,还可以使用Optional进行优化。 

@Test        public void shouldConvertDTO(){          PersonDTO personDTO = new PersonDTO();          Optional.ofNullable(getPerson()).ifPresent(person -> {            personDTO.setDtoAge(person.getAge());            personDTO.setDtoName(person.getName());          });        }        private Person getPerson(){          return null;        }

Optional对空值的使用,我觉得更为贴切,它只适用于”是否存在”的场景。

如果只对控制的存在判断,我建议使用Optional。

Optioanl的正确使用

Optional如此强大,它表达了计算机最原始的特性(0 or 1),那它如何正确的被使用呢!

Optional不要作为参数

如果你写了一个public方法,这个方法规定了一些输入参数,这些参数中有一些是可以传入null的,那这时候是否可以使用Optional呢?

给的建议是: 一定不要这样使用!

举个例子: 

public interface UserService{        List<User> listUser(Optional<String> username);      }

这个例子的方法 listUser,可能在告诉我们需要根据username查询所有数据集合,如果username是空,也要返回所有的用户集合.

当我们看到这个方法的时候,会觉得有一些歧义:

“如果username是absent,是返回空集合吗?还是返回全部的用户数据集合?”

Optioanl是一种分支的判断,那我们究竟是关注 Optional还是Optional.get()呢?

给大家的建议是,如果不想要这样的歧义,就不要使用它!

如果你真的想表达两个含义,就給它拆分出两个接口:

public interface UserService{      List<User> listUser(String username);      List<User> listUser();    }

我觉得这样的语义更强,并且更能满足 软件设计原则中的 “单一职责”。

如果你觉得你的入参真的有必要可能传null,那请使用jsr 303或者jsr 305进行说明和验证!

请记住! Optional不能作为入参的参数!

Optional作为返回值

当个实体的返回

那Optioanl可以做为返回值吗?

其实它是非常满足是否存在这个语义的。

你如说,你要根据id获取用户信息,这个用户有可能存在或者不存在。

你可以这样使用:

public interface UserService{     Optional<User> get(Integer id);   }

当调用这个方法的时候,调用者很清楚get方法返回的数据,有可能不存在,这样可以做一些更合理的判断,更好的防止空指针的错误!

当然,如果业务方真的需要根据id必须查询出User的话,就不要这样使用了,请说明,你要抛出的异常.

只有当考虑它返回null是合理的情况下,才进行Optional的返回

集合实体的返回

不是所有的返回值都可以这样用的!如果你返回的是集合:

public interface UserService{    Optional<List<User>> listUser();  }

这样的返回结果,会让调用者不知所措,是否我判断Optional之后,还用进行isEmpty的判断呢?

这样带来的返回值歧义!我认为是没有必要的。

我们要约定,对于List这种集合返回值,如果集合真的是null的,请返回空集合(Lists.newArrayList);

使用Optional变量 

Optional<User> userOpt = ...

如果有这样的变量userOpt,请记住 :

getter中的使用

对于一个java bean,所有的属性都有可能返回null,那是否需要改写所有的getter成为Optional类型呢?

给大家的建议是,不要这样滥用Optional.

即便 我java bean中的getter是符合Optional的,但是因为java bean 太多了,这样会导致你的代码有50%以上进行Optinal的判断,这样便污染了代码。(我想说,其实你的实体中的字段应该都是由业务含义的,会认真的思考过它存在的价值的,不能因为Optional的存在而滥用)

我们应该更关注于业务,而不只是空值的判断。

不要在getter中滥用Optional.

感谢各位的阅读,以上就是“Java中如何解决null值引起的Bug”的内容了,经过本文的学习后,相信大家对Java中如何解决null值引起的Bug这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是亿速云,小编将为大家推送更多相关知识点的文章,欢迎关注!

推荐阅读:
  1. MySQL-JDBC驱动引起bug问题的示例
  2. Mybatis整合Spring 由于版本引起的BUG问题

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

java

上一篇:Spring IOC初始化执行流程是什么

下一篇:Python为什么不支持i++自增语法

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》