1. 首页
  2. IT资讯

Java:如何更优雅的处理空值?

“u003Cpu003E优质文章,及时送达u003Cu002Fpu003Eu003Cimg src=”http:u002Fu002Fp1.pstatp.comu002Flargeu002Fpgc-imageu002FRcsUnd92eoboDR” img_width=”640″ img_height=”29″ alt=”Java:如何更优雅的处理空值?” inline=”0″u003Eu003Cpu003E来源 | lrwinxu003Cu002Fpu003Eu003Cpu003E作者 | https:u002Fu002Flrwinx.github.iou002Fu003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch1 toutiao-origin=”h2″u003E导语u003Cu002Fh1u003Eu003Cpu003E在笔者几年的开发经验中,经常看到项目中存在到处空值判断的情况,这些判断,会让人觉得摸不着头绪,它的出现很有可能和当前的业务逻辑并没有关系。但它会让你很头疼。u003Cu002Fpu003Eu003Cpu003E有时候,更可怕的是系统因为这些空值的情况,会抛出空指针异常,导致业务系统发生问题。u003Cu002Fpu003Eu003Cpu003E此篇文章,我总结了几种关于空值的处理手法,希望对读者有帮助。u003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch1 toutiao-origin=”h2″u003E业务中的空值u003Cu002Fh1u003Eu003Ch2 toutiao-origin=”h3″u003Eu003Cstrongu003E场景u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E存在一个UserSearchService用来提供用户查询的功能:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic interface UserSearchService{u003Cbru003EList<User> listUser;u003Cbru003Eu003Cbru003EUser get(Integer id);u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E问题现场u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E对于面向对象语言来讲,抽象层级特别的重要。尤其是对接口的抽象,它在设计和开发中占很大的比重,我们在开发时希望尽量面向接口编程。u003Cu002Fpu003Eu003Cpu003E对于以上描述的接口方法来看,大概可以推断出可能它包含了以下两个含义:u003Cu002Fpu003Eu003Culu003Eu003Cliu003Eu003Cpu003ElistUser: 查询用户列表u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003Eget(Integer id): 查询单个用户u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cu002Fulu003Eu003Cpu003E在所有的开发中,XP推崇的Tu003Ci class=”chrome-extension-mutihighlight chrome-extension-mutihighlight-style-3″u003EDDu003Cu002Fiu003E模式可以很好的引导我们对接口的定义,所以我们将Tu003Ci class=”chrome-extension-mutihighlight chrome-extension-mutihighlight-style-3″u003EDDu003Cu002Fiu003E作为开发代码的”推动者”。u003Cu002Fpu003Eu003Cpu003E对于以上的接口,当我们使用Tu003Ci class=”chrome-extension-mutihighlight chrome-extension-mutihighlight-style-3″u003EDDu003Cu002Fiu003E进行测试用例先行时,发现了潜在的问题:u003Cu002Fpu003Eu003Culu003Eu003Cliu003Eu003Cpu003ElistUser 如果没有数据,那它是返回空集合还是呢?u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003Eget(Integer id) 如果没有这个对象,是抛异常还是返回呢?u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cu002Fulu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E深入listUser研究u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E我们先来讨论u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003ElistUseru003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E这个接口,我经常看到如下实现:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic List<User> listUser{u003Cbru003EList<User> userList = userListRepostity.selectByExample(new UserExample);u003Cbru003Eif(CollectionUtils.isEmpty(userList)){u002Fu002Fspring util工具类u003Cbru003Ereturn ;u003Cbru003E}u003Cbru003Ereturn userList;u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E这段代码返回是,从我多年的开发经验来讲,对于集合这样返回值,最好不要返回,因为如果返回了,会给调用者带来很多麻烦。你将会把这种调用风险交给调用者来控制。u003Cu002Fpu003Eu003Cpu003E如果调用者是一个谨慎的人,他会进行是否为的条件判断。如果他并非谨慎,或者他是一个面向接口编程的狂热分子(当然,面向接口编程是正确的方向),他会按照自己的理解去调用接口,而不进行是否为的条件判断,如果这样的话,是非常危险的,它很有可能出现空指针异常!u003Cu002Fpu003Eu003Cpu003E根据墨菲定律来判断: u003Cstrongu003E“很有可能出现的问题,在将来一定会出现!”u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpu003E基于此,我们将它进行优化:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic List<User> listUser{u003Cbru003EList<User> userList = userListRepostity.selectByExample(new UserExample);u003Cbru003Eif(CollectionUtils.isEmpty(userList)){u003Cbru003Ereturn Lists.newArrayList;u002Fu002Fguava类库提供的方式u003Cbru003E}u003Cbru003Ereturn userList;u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E对于接口u003Ccodeu003Eu003Cstrong toutiao-origin=”span” class=”highlight-text”u003E(List listUser)u003Cu002Fstrongu003Eu003Cu002Fcodeu003E,它一定会返回List,即使没有数据,它仍然会返回List(集合中没有任何元素);u003Cu002Fpu003Eu003Cpu003E通过以上的修改,我们成功的避免了有可能发生的空指针异常,这样的写法更安全!u003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E深入研究get方法u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E对于接口u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003EUser get(Integer id)u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E你能看到的现象是,我给出id,它一定会给我返回User.但事实真的很有可能不是这样的。u003Cu002Fpu003Eu003Cpu003E我看到过的实现:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic User get(Integer id){u003Cbru003Ereturn userRepository.selectByPrimaryKey(id);u002Fu002F从数据库中通过id直接获取实体对象u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E相信很多人也都会这样写。u003Cu002Fpu003Eu003Cpu003E通过代码的时候得知它的返回值很有可能是! 但我们通过的接口是分辨不出来的!u003Cu002Fpu003Eu003Cpu003E这个是个非常危险的事情。尤其对于调用者来说!u003Cu002Fpu003Eu003Cpu003E我给出的建议是,需要在接口明明时补充文档,比如对于异常的说明,使用注解@exception:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic u003Cstrong toutiao-origin=”span”u003Einterfaceu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EUserSearchServiceu003Cu002Fstrongu003E{u003Cbru003Eu003Cbru003Eu002F**u003Cbru003E* 根据用户id获取用户信息u003Cbru003E* @param id 用户idu003Cbru003E* @return 用户实体u003Cbru003E* @exception UserNotFoundExceptionu003Cbru003E*u002Fu003Cbru003EUser get(Integer id);u003Cbru003Eu003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E我们把接口定义加上了说明之后,调用者会看到,如果调用此接口,很有可能抛出“UserNotFoundException(找不到用户)”这样的异常。u003Cu002Fpu003Eu003Cpu003E这种方式可以在调用者调用接口的时候看到接口的定义,但是,这种方式是”弱提示”的!u003Cu002Fpu003Eu003Cpu003E如果调用者忽略了注释,有可能就对业务系统产生了风险,这个风险有可能导致一个亿!u003Cu002Fpu003Eu003Cpu003Eu003Cstrongu003E除了以上这种”弱提示”的方式,还有一种方式是,返回值是有可能为空的。u003Cu002Fstrongu003Eu003Cstrongu003E那要怎么办呢?u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpu003E我认为我们需要增加一个接口,用来描述这种场景.u003Cu002Fpu003Eu003Cpu003E引入jdk8的Optional,或者使用guava的Optional.看如下定义:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic u003Cstrong toutiao-origin=”span”u003Einterfaceu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EUserSearchServiceu003Cu002Fstrongu003E{u003Cbru003Eu003Cbru003Eu002F**u003Cbru003E* 根据用户id获取用户信息u003Cbru003E* @param id 用户idu003Cbru003E* @return 用户实体,此实体有可能是缺省值u003Cbru003E*u002Fu003Cbru003EOptional<User> getOptional(Integer id);u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003EOptional有两个含义: 存在 or 缺省。u003Cu002Fpu003Eu003Cpu003E那么通过阅读接口getOptional,我们可以很快的了解返回值的意图,这个其实是我们想看到的,它去除了二义性。u003Cu002Fpu003Eu003Cpu003E它的实现可以写成:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic Optional<User> getOptional(Integer id){u003Cbru003Ereturn Optional.ofable(userRepository.selectByPrimaryKey(id));u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E深入入参u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E通过上述的所有接口的描述,你能确定入参id一定是必传的吗?我觉得答案应该是:不能确定。除非接口的文档注释上加以说明。u003Cu002Fpu003Eu003Cpu003Eu003Cstrongu003E那如何约束入参呢?u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpu003E我给大家推荐两种方式:u003Cu002Fpu003Eu003Culu003Eu003Cliu003Eu003Cpu003E强制约束u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003E文档性约束(弱提示)u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cu002Fulu003Eu003Cpu003Eu003Cstrongu003E1.强制约束,我们可以通过jsr 303进行严格的约束声明:u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic u003Cstrong toutiao-origin=”span”u003Einterfaceu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EUserSearchServiceu003Cu002Fstrongu003E{u003Cbru003Eu002F**u003Cbru003E* 根据用户id获取用户信息u003Cbru003E* @param id 用户idu003Cbru003E* @return 用户实体u003Cbru003E* @exception UserNotFoundExceptionu003Cbru003E*u002Fu003Cbru003EUser get(@Not Integer id);u003Cbru003Eu003Cbru003Eu002F**u003Cbru003E* 根据用户id获取用户信息u003Cbru003E* @param id 用户idu003Cbru003E* @return 用户实体,此实体有可能是缺省值u003Cbru003E*u002Fu003Cbru003EOptional<User> getOptional(@Not Integer id);u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E当然,这样写,要配合AOP的操作进行验证,但让spring已经提供了很好的集成方案,在此我就不在赘述了。u003Cu002Fpu003Eu003Cpu003Eu003Cstrongu003E2.文档性约束u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpu003E在很多时候,我们会遇到遗留代码,对于遗留代码,整体性改造的可能性很小。u003Cu002Fpu003Eu003Cpu003E我们更希望通过阅读接口的实现,来进行接口的说明。u003Cu002Fpu003Eu003Cpu003Ejsr 305规范,给了我们一个描述接口入参的一个方式(需要引入库 com.google.code.findbugs:jsr305):u003Cu002Fpu003Eu003Cpu003E可以使用注解: @able @Non @CheckFor 进行接口说明。比如:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic u003Cstrong toutiao-origin=”span”u003Einterfaceu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EUserSearchServiceu003Cu002Fstrongu003E{u003Cbru003Eu002F**u003Cbru003E* 根据用户id获取用户信息u003Cbru003E* @param id 用户idu003Cbru003E* @return 用户实体u003Cbru003E* @exception UserNotFoundExceptionu003Cbru003E*u002Fu003Cbru003E@CheckForu003Cbru003EUser get(@Non Integer id);u003Cbru003Eu003Cbru003Eu002F**u003Cbru003E* 根据用户id获取用户信息u003Cbru003E* @param id 用户idu003Cbru003E* @return 用户实体,此实体有可能是缺省值u003Cbru003E*u002Fu003Cbru003EOptional<User> getOptional(@Non Integer id);u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrong toutiao-origin=”span” class=”highlight-text”u003E小结u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E通过 空集合返回值,Optional,jsr 303,jsr 305这几种方式,可以让我们的代码可读性更强,出错率更低!u003Cu002Fpu003Eu003Culu003Eu003Cliu003Eu003Cpu003E空集合返回值 :如果有集合这样返回值时,除非真的有说服自己的理由,否则,一定要返回空集合,而不是u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003EOptional: 如果你的代码是jdk8,就引入它!如果不是,则使用Guava的Optional,或者升级jdk版本!它很大程度的能增加了接口的可读性!u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003Ejsr 303: 如果新的项目正在开发,不防加上这个试试!一定有一种特别爽的感觉!u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003Ejsr 305: 如果老的项目在你的手上,你可以尝试的加上这种文档型注解,有助于你后期的重构,或者新功能增加了,对于老接口的理解!u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cu002Fulu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch1 toutiao-origin=”h2″u003E空对象模式u003Cu002Fh1u003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E场景u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E我们来看一个DTO转化的场景,对象:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003E@Datau003Cbru003Estatic class PersonDTO{u003Cbru003Eprivate String dtoName;u003Cbru003Eprivate String dtoAge;u003Cbru003E}u003Cbru003Eu003Cbru003E@Datau003Cbru003Estatic class Person{u003Cbru003Eprivate String name;u003Cbru003Eprivate String age;u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E需求是将Person对象转化成PersonDTO,然后进行返回。u003Cu002Fpu003Eu003Cpu003E当然对于实际操作来讲,返回如果Person为空,将返回,但是PersonDTO是不能返回的(尤其Rest接口返回的这种DTO)。u003Cu002Fpu003Eu003Cpu003E在这里,我们只u003Ci class=”chrome-extension-mutihighlight chrome-extension-mutihighlight-style-2″u003E关注u003Cu002Fiu003E转化操作,看如下代码:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Eu003Cstrong toutiao-origin=”span”u003E@Testu003Cu002Fstrongu003Eu003Cbru003Eu003Cstrong toutiao-origin=”span”u003Epublicu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003Evoidu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EshouldConvertDTOu003Cu002Fstrongu003E{u003Cbru003Eu003Cbru003EPersonDTO personDTO = new PersonDTO;u003Cbru003Eu003Cbru003EPerson person = new Person;u003Cbru003Eif(!Objects.is(person)){u003Cbru003EpersonDTO.setDtoAge(person.getAge);u003Cbru003EpersonDTO.setDtoName(person.getName);u003Cbru003E}else{u003Cbru003EpersonDTO.setDtoAge(“”);u003Cbru003EpersonDTO.setDtoName(“”);u003Cbru003E}u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E优化修改u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E这样的数据转化,我们认识可读性非常差,每个字段的判断,如果是空就设置为空字符串(“”)u003Cu002Fpu003Eu003Cpu003E换一种思维方式进行思考,我们是拿到Person这个类的数据,然后进行赋值操作(setXXX),其实是不关系Person的具体实现是谁的。u003Cu002Fpu003Eu003Cpu003E那我们可以创建一个Person子类:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Estatic u003Cstrong toutiao-origin=”span”u003Eclassu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EPersonu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003Eextendsu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EPersonu003Cu002Fstrongu003E{u003Cbru003Eu003Cstrong toutiao-origin=”span”u003E@Overrideu003Cu002Fstrongu003Eu003Cbru003Epublic String getAge {u003Cbru003Ereturn “”;u003Cbru003E}u003Cbru003Eu003Cbru003Eu003Cstrong toutiao-origin=”span”u003E@Overrideu003Cu002Fstrongu003Eu003Cbru003Epublic String getName {u003Cbru003Ereturn “”;u003Cbru003E}u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E它作为Person的一种特例而存在,如果当Person为空的时候,则返回一些get*的默认行为.u003Cu002Fpu003Eu003Cpu003E所以代码可以修改为:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003E@Testu003Cbru003Eu003Cstrong toutiao-origin=”span”u003Epublicu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003Evoidu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EshouldConvertDTOu003Cu002Fstrongu003E{u003Cbru003Eu003Cbru003EPersonDTO personDTO = new PersonDTO;u003Cbru003Eu003Cbru003EPerson person = getPerson;u003Cbru003EpersonDTO.setDtoAge(person.getAge);u003Cbru003EpersonDTO.setDtoName(person.getName);u003Cbru003E}u003Cbru003Eu003Cbru003Eprivate Person getPerson{u003Cbru003Eu003Cstrong toutiao-origin=”span”u003Eru003Cu002Fstrongu003Eeturn new Person;u002Fu002F如果Person是 ,则返回空对象u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E其中getPerson方法,可以用来根据业务逻辑获取Person有可能的对象(对当前例子来讲,如果Person不存在,返回Person的的特例Person),如果修改成这样,代码的可读性就会变的很强了。u003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E使用Optional可以进行优化u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E空对象模式,它的弊端在于需要创建一个特例对象,但是如果特例的情况比较多,我们是不是需要创建多个特例对象呢,虽然我们也使用了面向对象的多态特性,但是,业务的复杂性如果真的让我们创建多个特例对象,我们还是要再三考虑一下这种模式,它可能会带来代码的复杂性。u003Cu002Fpu003Eu003Cpu003E对于上述代码,还可以使用Optional进行优化。u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Eu003Cstrong toutiao-origin=”span”u003E@Testu003Cu002Fstrongu003Eu003Cbru003Eu003Cstrong toutiao-origin=”span”u003Epublicu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003Evoidu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EshouldConvertDTOu003Cu002Fstrongu003E{u003Cbru003Eu003Cbru003EPersonDTO personDTO = new PersonDTO;u003Cbru003Eu003Cbru003EOptional.ofable(getPerson).ifPresent(person -> {u003Cbru003EpersonDTO.setDtoAge(person.getAge);u003Cbru003EpersonDTO.setDtoName(person.getName);u003Cbru003E});u003Cbru003E}u003Cbru003Eu003Cbru003Eprivate Person getPerson{u003Cbru003Ereturn ;u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003EOptional对空值的使用,我觉得更为贴切,它只适用于”是否存在”的场景。u003Cu002Fpu003Eu003Cpu003E如果只对控制的存在判断,我建议使用Optional.u003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003EOptioanl的正确使用u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003EOptional如此强大,它表达了计算机最原始的特性(0 or 1),那它如何正确的被使用呢!u003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003EOptional不要作为参数u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E如果你写了一个public方法,这个方法规定了一些输入参数,这些参数中有一些是可以传入的,那这时候是否可以使用Optional呢?u003Cu002Fpu003Eu003Cpu003Eu003Cstrongu003E我给的建议是: 一定不要这样使用!u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpu003E举个例子:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic u003Cstrong toutiao-origin=”span”u003Einterfaceu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EUserServiceu003Cu002Fstrongu003E{u003Cbru003EList<User> listUser(Optional<String> username);u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E这个例子的方法 listUser,可能在告诉我们需要根据username查询所有数据集合,如果username是空,也要返回所有的用户集合.u003Cu002Fpu003Eu003Cpu003E当我们看到这个方法的时候,会觉得有一些歧义:u003Cu002Fpu003Eu003Cblockquoteu003Eu003Cpu003E“如果username是absent,是返回空集合吗?还是返回全部的用户数据集合?”u003Cu002Fpu003Eu003Cu002Fblockquoteu003Eu003Cpu003EOptioanl是一种分支的判断,那我们究竟是u003Ci class=”chrome-extension-mutihighlight chrome-extension-mutihighlight-style-2″u003E关注u003Cu002Fiu003E Optional还是Optional.get呢?u003Cu002Fpu003Eu003Cpu003Eu003Cstrongu003E我给大家的建议是,如果不想要这样的歧义,就不要使用它!u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpu003E如果你真的想表达两个含义,就給它拆分出两个接口:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic u003Cstrong toutiao-origin=”span”u003Einterfaceu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EUserServiceu003Cu002Fstrongu003E{u003Cbru003EList<User> listUser(String username);u003Cbru003EList<User> listUser;u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E我觉得这样的语义更强,并且更能满足 软件设计原则中的 “单一职责”。u003Cu002Fpu003Eu003Cpu003E如果你觉得你的入参真的有必要可能传,那请使用jsr 303或者jsr 305进行说明和验证!u003Cu002Fpu003Eu003Cpu003Eu003Cstrongu003E请记住! Optional不能作为入参的参数!u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch1 toutiao-origin=”h2″u003EOptional作为返回值u003Cu002Fh1u003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E当个实体的返回u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E那Optioanl可以做为返回值吗?u003Cu002Fpu003Eu003Cpu003E其实它是非常满足是否存在这个语义的。u003Cu002Fpu003Eu003Cpu003E你如说,你要根据id获取用户信息,这个用户有可能存在或者不存在。u003Cu002Fpu003Eu003Cpu003E你可以这样使用:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic interface UserService{u003Cbru003EOptional<User> get(Integer id);u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E当调用这个方法的时候,调用者很清楚get方法返回的数据,有可能不存在,这样可以做一些更合理的判断,更好的防止空指针的错误!u003Cu002Fpu003Eu003Cpu003E当然,如果业务方真的需要根据id必须查询出User的话,就不要这样使用了,请说明,你要抛出的异常.u003Cu002Fpu003Eu003Cpu003E只有当考虑它返回是合理的情况下,才进行Optional的返回u003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003E集合实体的返回u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E不是所有的返回值都可以这样用的!如果你返回的是集合:u003Cu002Fpu003Eu003Cpreu003Eu003Ccodeu003Epublic u003Cstrong toutiao-origin=”span”u003Einterfaceu003Cu002Fstrongu003Eu003Cstrong toutiao-origin=”span”u003EUserServiceu003Cu002Fstrongu003E{u003Cbru003EOptional<List<User>> listUser;u003Cbru003E}u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E这样的返回结果,会让调用者不知所措,是否我判断Optional之后,还用进行isEmpty的判断呢?u003Cu002Fpu003Eu003Cpu003E这样带来的返回值歧义!我认为是没有必要的。u003Cu002Fpu003Eu003Cpu003E我们要约定,对于List这种集合返回值,如果集合真的是的,请返回空集合(Lists.newArrayList);u003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch1 toutiao-origin=”h2″u003E使用Optional变量u003Cu002Fh1u003Eu003Cpreu003Eu003Ccodeu003EOptional<User> userOpt = …u003Cu002Fcodeu003Eu003Cu002Fpreu003Eu003Cpu003E如果有这样的变量userOpt,请记住 :u003Cu002Fpu003Eu003Culu003Eu003Cliu003Eu003Cpu003E一定不能直接使用get ,如果这样用,就丧失了Optional本身的含义 ( 比如userOp.get() )u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003E不要直接使用getOrThrow ,如果你有这样的需求:获取不到就抛异常。那就要考虑,是否是调用的接口设计的是否合理u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cu002Fulu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrongu003Egetter中的使用u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E对于一个java bean,所有的属性都有可能返回,那是否需要改写所有的getter成为Optional类型呢?u003Cu002Fpu003Eu003Cpu003Eu003Cstrongu003E我给大家的建议是,不要这样滥用Optional.u003Cu002Fstrongu003Eu003Cu002Fpu003Eu003Cpu003E即便 我java bean中的getter是符合Optional的,但是因为java bean 太多了,这样会导致你的代码有50%以上进行Optinal的判断,这样便污染了代码。(我想说,其实你的实体中的字段应该都是由业务含义的,会认真的思考过它存在的价值的,不能因为Optional的存在而滥用)u003Cu002Fpu003Eu003Cpu003E我们应该更u003Ci class=”chrome-extension-mutihighlight chrome-extension-mutihighlight-style-2″u003E关注u003Cu002Fiu003E于业务,而不只是空值的判断。u003Cu002Fpu003Eu003Cpu003E请不要在getter中滥用Optional.u003Cu002Fpu003Eu003Cpu003Eu003Cu002Fpu003Eu003Ch2 toutiao-origin=”h4″u003Eu003Cstrong toutiao-origin=”span” class=”highlight-text”u003E小结u003Cu002Fstrongu003Eu003Cu002Fh2u003Eu003Cpu003E可以这样总结Optional的使用:u003Cu002Fpu003Eu003Culu003Eu003Cliu003Eu003Cpu003E当使用值为空的情况,并非源于错误时,可以使用Optional!u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003EOptional不要用于集合操作!u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cliu003Eu003Cpu003E不要滥用Optional,比如在java bean的getter中!u003Cu002Fpu003Eu003Cu002Fliu003Eu003Cu002Fulu003Eu003Cp class=”pgc-end-source”u003E【END】u003Cu002Fpu003E”

原文始发于:Java:如何更优雅的处理空值?

主题测试文章,只做测试使用。发布者:逗乐男神i,转转请注明出处:http://www.cxybcw.com/17748.html

联系我们

13687733322

在线咨询:点击这里给我发消息

邮件:1877088071@qq.com

工作时间:周一至周五,9:30-18:30,节假日休息

QR code