optional 解决空指针_实战系列:使用Java8 Optional类优雅解决空指针问题

原标题:实战系列:使用Java8 Optional类优雅解决空指针问题

关注 “”

作者 | 雷架

来源 | 爱笑的架构师(ID:DancingOnYourCode)

Java8 由Oracle在2014年发布,是继Java5之后最具革命性的版本。Java8吸收其他语言的精髓带来了函数式编程,lambda表达式,Stream流等一系列新特性,学会了这些新特性,可以让你实现高效编码优雅编码。1. 不受待见的空指针异常

有个小故事:null引用最早是由英国科学家Tony Hoare提出的,多年后Hoare为自己的这个想法感到后悔莫及,并认为这是"价值百万的重大失误"。可见空指针是多么不受待见。

NullPointerException是Java开发中最常遇见的异常,遇到这种异常我们通常的解决方法是在调用的地方加一个if判空。

if判空越多会造成过多的代码分支,后续代码维护也就越来越复杂。

2. 糟糕的代码

比如看下面这个例子,使用过多的if判空。

Person对象里定义了House对象,House对象里定义了Address对象:

publicclassPerson{

privateString name;

privateintage;

privateHouse house;

publicHouse getHouse{

returnhouse;

}

}

classHouse{

privatelongprice;

privateAddress address;

publicAddress getAddress{

returnaddress;

}

}

classAddress{

privateString country;

privateString city;

publicString getCity{

returncity;

}

}

现在获取这个人买房的城市,那么通常会这样写:

publicString getCity{

String city = newPerson.getHouse.getAddress.getCity;

returncity;

}

但是这样写容易出现空指针的问题,比如这个人没有房,House对象为null。接着你会改造这段代码,加上很多判断条件:

publicString getCity2(Person person){

if(person != null) {

House house = person.getHouse;

if(house != null) {

Address address = house.getAddress;

if(address != null) {

String city = address.getCity;

returncity;

}

}

}

return"unknown";

}

为了避免空指针异常,每一层都加上判断,但是这样会造成代码嵌套太深,不易维护。你可能想到如何改造上面的代码,比如加上提前判空退出:

publicString getCity3(Person person){

String city = "unknown";

if(person == null) {

returncity;

}

House house = person.getHouse;

if(house == null) {

returncity;

}

Address address = house.getAddress;

if(address == null) {

returncity;

}

returnaddress.getCity;

}

但是这样简单的代码已经加入了三个退出条件,非常不利于后面代码维护。那怎样才能将代码写的优雅一点呢,下面引入今天的主角"Optional"。

3. 解决空指针的"银弹"

从Java8开始引入了一个新类 java.util.Optional,这是一个对象的容器,意味着可能包含或者没有包含一个非空的值。下面重点看一下Optional的常用方法:

publicfinalclassOptional< T>{

// 通过指定非空值创建Optional对象

// 如果指定的值为null,会抛空指针异常

publicstatic Optional of(T value){

returnnewOptional<>(value);

}

// 通过指定可能为空的值创建Optional对象

publicstatic Optional ofNullable(T value){

returnvalue == null? empty : of(value);

}

// 返回值,不存在抛异常

publicT get{

if(value == null) {

thrownewNoSuchElementException( "No value present");

}

returnvalue;

}

// 如果值存在,根据consumer实现类消费该值

publicvoidifPresent(Consumer superT> consumer){

if(value != null)

consumer.accept(value);

}

// 如果值存在则返回,如果值为空则返回指定的默认值

publicT orElse(T other){

returnvalue != null? value : other;

}

// map flatmap等方法与Stream使用方法类似,这里不再赘述,读者可以参考之前的Stream系列。

}

以上就是Optional类常用的方法,使用起来非常简单。

4. Optional使用入门

(1)创建Optional实例

创建空的Optional对象。可以通过静态工厂方法Optional.Empty 创建一个空的对象,例如:Optional optionalPerson = Optional.Empty;

指定非空值创建Optional对象。Person person =

newPerson;

Optional optionalPerson = Optional.of(person);

指定可能为空的值创建Optional对象。Person person =

null;

// 可能为空

Optional optionalPerson = Optional.of(person);

(2)常用方法ifPresent

如果值存在,则调用consumer实例消费该值,否则什么都不执行。举个栗子:

String str = "hello java8";

// output: hello java8

Optional.ofNullable(str).ifPresent(System.out::println);

String str2 = null;

// output: nothing

Optional.ofNullable(str2).ifPresent(System.out::println);

filter, map, flatMap在三个方法在前面讲Stream的时候已经详细讲解过,读者可以翻看之前写的文章,这里不再赘述。

orElse

如果value为空,则返回默认值,举个栗子:

publicvoidtest(String city){

String defaultCity = Optional.ofNullable(city).orElse( "unknown");

}

orElseGet如果value为空,则调用Supplier实例返回一个默认值。举个例子:

publicvoidtest2(String city){

// 如果city为空,则调用generateDefaultCity方法

String defaultCity = Optional.of(city).orElseGet( this::generateDefaultCity);

}

privateString generateDefaultCity{

return"beijing";

}

orElseThrow如果value为空,则抛出自定义异常。举个栗子:

publicvoidtest3(String city){

// 如果city为空,则抛出空指针异常。

String defaultCity = Optional.of(city).orElseThrow(NullPointerException:: new);

}

5. 使用Optional重构代码

再看一遍重构之前的代码,使用了三个if使代码嵌套层次变得很深。

// before refactor

publicString getCity2(Person person){

if(person != null) {

House house = person.getHouse;

if(house != null) {

Address address = house.getAddress;

if(address != null) {

String city = address.getCity;

returncity;

}

}

}

return"unknown";

}

使用Optional重构

publicString getCityUsingOptional(Person person){

String city = Optional.ofNullable(person)

.map(Person::getHouse)

.map(House::getAddress)

.map(Address::getCity).orElse( "Unknown city");

returncity;

}

只使用了一行代码就获取到city值,不用再去不断的判断是否为空,这样写代码是不是很优雅呀。

总结:使用optional类可以很优雅的解决项目中空指针的问题,但是optional也不是万能的哦,小伙伴们要适度使用。赶紧用Optional重构之前写的项目吧~返回搜狐,查看更多

责任编辑:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值