对于面向对象语言来讲,抽象层级特别的重要。尤其是对接口的抽象,它在设计和开发中占很大的比重,我们在开发时希望尽量面向接口编程。对于以上描述的接口方法来看,大概可以推断出可能它包含了以下两个含义: 1. listUser(): 查询用户列表 2. get(Integer id): 查询单个用户 在所有的开发中,XP推崇的TDD模式可以很好的引导我们对接口的定义,所以我们将TDD作为开发代码的”推动者”。对于以上的接口,当我们使用TDD进行测试用例先行时,发现了潜在的问题: 1. listUser() 如果没有数据,那它是返回空集合还是null呢? 2. get(Integer id) 如果没有这个对象,是抛异常还是返回null呢? ### 深入listUser研究 1. 来看一下listUser() 这个列表 ``` public List listUser(){      List userList = userListRepostity.selectByExample(new UserExample()); if(CollectionUtils.isEmpty(userList)){ //spring util工具类 returnnull;      } return userList; } ``` > 这段代码返回是null,从开发经验来讲,对于集合这样返回值,最好不要返回null,因为如果返回了null,会给调用者带来很多麻烦。你将会把这种调用风险交给调用者来控制。如果调用者是一个谨慎的人,他会进行是否为null的条件判断。如果他并非谨慎,或者他是一个面向接口编程的狂热分子(当然,面向接口编程是正确的方向),他会按照自己的理解去调用接口,而不进行是否为null的条件判断,如果这样的话,是非常危险的,它很有可能出现空指针异常!根据墨菲定律来判断: “很有可能出现的问题,在将来一定会出现!” ### 基于此,我们将它进行优化: ``` public List listUser(){      List userList = userListRepostity.selectByExample(new UserExample()); if(CollectionUtils.isEmpty(userList)){ return Lists.newArrayList();//guava类库提供的方式      } return userList; } ``` 对于接口(List listUser()),它一定会返回List,即使没有数据,它仍然会返回List(集合中没有任何元素); 通过以上的修改,我们成功的避免了有可能发生的空指针异常,这样的写法更安全! ### 深入研究get方法 对于接口 ``` User get(Integer id) ``` 你能看到的现象是,我给出id,它一定会给我返回User.但事实真的很有可能不是这样的。 我看到过的实现: ``` public User get(Integer id){ //从数据库中通过id直接获取实体对象 return userRepository.selectByPrimaryKey(id); } ``` 相信很多人也都会这样写。通过代码的时候得知它的返回值很有可能是null! 但我们通过的接口是分辨不出来的! 这个是个非常危险的事情。尤其对于调用者来说! 我给出的建议是,需要在接口明明时补充文档,比如对于异常的说明,使用注解@exception: ``` publicinterface UserSearchService{ /**    * 根据用户id获取用户信息    * @param id 用户id    * @return 用户实体    * @exception UserNotFoundException    */ User get(Integer id); } ``` 我们把接口定义加上了说明之后,调用者会看到,如果调用此接口,很有可能抛出“UserNotFoundException(找不到用户)”这样的异常。 这种方式可以在调用者调用接口的时候看到接口的定义,但是,这种方式是”弱提示”的!如果调用者忽略了注释,有可能就对业务系统产生了风险,这个风险有可能导致一个亿! 除了以上这种”弱提示”的方式,还有一种方式是,返回值是有可能为空的。那要怎么办呢?我认为我们需要增加一个接口,用来描述这种场景. 引入jdk8的Optional,或者使用guava的Optional.看如下定义: ``` publicinterface UserSearchService{ /**  * 根据用户id获取用户信息  * @param id 用户id  * @return 用户实体,此实体有可能是缺省值  */ Optional getOptional(Integer id); } ``` Optional有两个含义: 存在 or 缺省。 那么通过阅读接口getOptional(),我们可以很快的了解返回值的意图,这个其实是我们想看到的,它去除了二义性。 它的实现可以写成: ``` public Optional getOptional(Integer id){ return Optional.ofNullable(userRepository.selectByPrimaryKey(id)); } ``` ### 深入入参 通过上述的所有接口的描述,你能确定入参id一定是必传的吗?我觉得答案应该是:不能确定。除非接口的文档注释上加以说明。 那如何约束入参呢? 我给大家推荐两种方式: 1. 强制约束 2. 文档性约束(弱提示) 1.强制约束,我们可以通过jsr 303进行严格的约束声明: ``` publicinterface UserSearchService{ /**   * 根据用户id获取用户信息   * @param id 用户id   * @return 用户实体   * @exception UserNotFoundException   */ User get(@NotNull Integer id); /**   * 根据用户id获取用户信息   * @param id 用户id   * @return 用户实体,此实体有可能是缺省值   */ Optional 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 用户实体 * @throws UserNotFoundException */ @CheckForNull User get(@NonNull Integer id); /** * 根据用户id获取用户信息 * * @param id 用户id * @return 用户实体, 此实体有可能是缺省值 */ Optional<User> getOptional(@NonNull Integer id); } ```