热头条丨Java8特性之Optional:如何干掉空指针?

2023-05-31 11:05:09 来源:个人图书馆-昵称10087950 分享到:

后台回复“大礼包”有惊喜礼包!


(资料图片)

每日英文

What is adhere to? Is day, and one day, you tell yourself, insist again one day.

什么是坚持?就是一天,又一天,你告诉自己,再坚持一天。

每日掏心话

不要沉迷过去,不要害怕未来,过去。得失也好,成败也罢,无论快乐,还是痛苦,都过去了,你只能回忆,而无法回去。

责编:乐乐 | 来自:blog.csdn.net/weixin_43888891/article/details/124788806

编程技术圈(ID:study_tech)第 1917期推文

往日回顾:当我去了不到 20 人的 IT 公司后,真的是大开眼界。。。

正文

大家好,我是小乐。

Optional的作用是什么?他都有哪些方法?阿里规范点名说尽量用Optional来避免空指针,那么什么场景用Optional?本篇文章围绕这三点来进行讲解。

目录

一、Optional类的来源

二、Optional类是什么?

三、Optional类用法

四、代码示例

1、创建Optional类

2、判断Optional容器中是否包含对象

3、获取Optional容器的对象

4、过滤

5、映射

五、什么场景用Optional?

1、场景一

2、场景二

3、场景三

4、场景四

一、Optional类的来源

到目前为止,臭名昭著的空指针异常是导致Java应用程序失败的最常见原因。以前,为了解决空指针异常,Google公司著名的Guava项目引入了Optional类,Guava通过使用检查空值的方式来防止代码污染,它鼓励程序员写更干净的代码。受到Google Guava的启发,Optional类已经成为Java 8类库的一部分。

二、Optional类是什么?

Optional 类(java.util.Optional) 是一个容器类,它可以保存类型T的值,代表这个值存在。或者仅仅保存null,表示这个值不存在。原来用 null 表示一个值不存在,现在 Optional 可以更好的表达这个概念。并且可以避免空指针异常。

Optional提供很多有用的方法,这样我们就不用显式进行空值检测。

三、Optional类用法

Optional类的Javadoc描述如下:这是一个可以为null的容器对象。

如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象。

如果值不存在则isPresent()方法会返回false,调用get()方法会NPE。

创建Optional类对象的方法:

Optional.of(T t): 创建一个 Optional 实例,t必须非空;

Optional.empty(): 创建一个空的 Optional 实例

Optional.ofNullable(T t):t可以为null

判断Optional容器中是否包含对象:

boolean isPresent(): 判断是否包含对象

void ifPresent(Consumer consumer):如果有值,就执行Consumer接口的实现代码,并且该值会作为参数传给它。

获取Optional容器的对象:

T get(): 如果调用对象包含值,返回该值,否则抛异常

T orElse(T other):如果有值则将其返回,否则返回指定的other对象。

T orElseGet(Supplier other):如果有值则将其返回,否则返回由Supplier接口实现提供的对象。

T orElseThrow(Supplier exceptionSupplier):如果有值则将其返回,否则抛出由Supplier接口实现提供的异常。

过滤:

Optional filter(Predicate predicate):如果值存在,并且这个值匹配给定的 predicate,返回一个Optional用以描述这个值,否则返回一个空的Optional。

映射

Optional map(Function mapper):如果有值,则对其执行调用映射函数得到返回值。如果返回值不为 null,则创建包含映射返回值的Optional作为map方法返回值,否则返回空Optional。

Optional flatMap(Function> mapper):如果值存在,就对该值执行提供的mapping函数调用,返回一个Optional类型的值,否则就返回一个空的Optional对象

四、代码示例

@Data@AllArgsConstructor@NoArgsConstructorclassStudent{privateStringname;privateIntegerage;}

1、创建Optional类

publicvoidtest1(){//声明一个空OptionalOptionalempty=Optional.empty();//依据一个非空值创建OptionalStudentstudent=newStudent();Optionalos1=Optional.of(student);//可接受null的OptionalStudentstudent1=null;Optionalos2=Optional.ofNullable(student1);}

2、判断Optional容器中是否包含对象

isPresent不带参数,判断是否为空,ifPresent可以选择带一个消费函数的实例。(isPresent和ifPresent一个是 is 一个是 if 注意一下哈)

publicvoidtest1(){Studentstudent=newStudent();Optionalos1=Optional.ofNullable(student);booleanpresent=os1.isPresent();System.out.println(present);//利用Optional的ifPresent方法做出如下:当student不为空的时候将name赋值为张三Optional.ofNullable(student).ifPresent(p->p.setName("张三"));}

3、获取Optional容器的对象

publicvoidtest1()throwsException{Studentstudent=null;Optionalos1=Optional.ofNullable(student);//使用get一定要注意,假如student对象为空,get是会报错的//java.util.NoSuchElementException:NovaluepresentStudentstudent1=os1.get();//当student为空的时候,返回我们新建的这个对象,有点像三目运算的感觉Studentstudent2=os1.orElse(newStudent("张三",18));//orElseGet就是当student为空的时候,返回通过Supplier供应商函数创建的对象Studentstudent3=os1.orElseGet(()->newStudent("张三",18));//orElseThrow就是当student为空的时候,可以抛出我们指定的异常os1.orElseThrow(()->newException());}

4、过滤

publicvoidtest1(){Studentstudent=newStudent("李四",3);Optionalos1=Optional.ofNullable(student);os1.filter(p->p.getName().equals("张三")).ifPresent(x->System.out.println("OK"));}

5、映射

map代码示例:

publicvoidtest1(){Studentstudent=newStudent("李四",3);Optionalos1=Optional.ofNullable(student);//如果student对象不为空,就加一岁Optionalemp=os1.map(e->{e.setAge(e.getAge()+1);returne;});}

这块的map说实话对lambda不是很熟练的 理解起来是很绕脑子的。

这里的map实际上就是用的Function函数,Function函数是有两个参数的,第一个是入参数据类型,第二个是返回数据类型。Function函数作用就是传入一个对象,然后返回一个对象,返回的对象类型可以自己设置。

T 就是代表实例的泛型数据类型,就是谁调用的 入参 必须跟调用者泛型的数据类型一样。

U 就是自己说了算,调用完map之后返回什么数据类型,那么U就设置什么

flatMap代码示例: flatMap跟map是一样的只不过他返回的是optional对象。

publicstaticOptionalstringToInt(Strings){try{returnOptional.of(Integer.parseInt(s));}catch(NumberFormatExceptione){e.printStackTrace();returnOptional.empty();}}
Optional.ofNullable(props.getProperty(name)).flatMap(OptionalUtils::stringToInt).filter(i->i>0).orElse(0);

五、什么场景用Optional?

以前一直不懂Optional有啥用,感觉太无语了,Java8还把它当做一个噱头来宣传,最近终于发现它的用处了,当然不用函数式编程的话,是没感觉的;

如下提供了几个应用场景,基本上都是开发当中经常遇到的。

1、场景一

PatientInfopatientInfo=patientInfoDao.getPatientInfoById(consultOrder.getPatientId());if(patientInfo!=null){consultInfoResp.setPatientHead(patientInfo.getHead());}//使用Optional和函数式编程,一行搞定,而且像说话一样Optional.ofNullable(patientInfo).ifPresent(p->consultInfoResp.setPatientHead(p.getHead()));

2、场景二

publicvoidtest1()throwsException{Studentstudent=newStudent(null,3);if(student==null||isEmpty(student.getName())){thrownewException();}Stringname=student.getName();//业务省略...//使用Optional改造Optional.ofNullable(student).filter(s->!isEmpty(s.getName())).orElseThrow(()->newException());}publicstaticbooleanisEmpty(CharSequencestr){returnstr==null||str.length()==0;}

3、场景三

publicstaticStringgetChampionName(Competitioncomp)throwsIllegalArgumentException{if(comp!=null){CompResultresult=comp.getResult();if(result!=null){Userchampion=result.getChampion();if(champion!=null){returnchampion.getName();}}}thrownewIllegalArgumentException("Thevalueofparamcompisn"tavailable.");}

这个在开发中是很常见的一种逻辑。去判读传进来的参数时候为空,或者是从数据库中获取的对象。由于某些原因,我们不能很流程的直接这样写。

comp.getResult().getChampion().getName()

上面的写法用Optional改写:

publicstaticStringgetChampionName(Competitioncomp)throwsIllegalArgumentException{returnOptional.ofNullable(comp).map(Competition::getResult)//相当于c->c.getResult(),下同.map(CompResult::getChampion).map(User::getName).orElseThrow(()->newIllegalArgumentException("Thevalueofparamcompisn"tavailable."));}

4、场景四

类型之间的转换,并且当没有值的时候返回一个默认值

inttimeout=Optional.ofNullable(redisProperties.getTimeout()).map(x->Long.valueOf(x.toMillis()).intValue()).orElse(10000);

为了跟上AI时代我干了一件事儿,我创建了一个知识星球社群:ChartGPT与副业。想带着大家一起探索ChatGPT和新的AI时代。

有很多小伙伴搞不定ChatGPT账号,于是我们决定,凡是这三天之内加入ChatPGT的小伙伴,我们直接送一个正常可用的永久ChatGPT独立账户。

不光是增长速度最快,我们的星球品质也绝对经得起考验,短短一个月时间,我们的课程团队发布了8个专栏、12个副业项目:

简单说下这个星球能给大家提供什么:

1、不断分享如何使用ChatGPT来完成各种任务,让你更高效地使用ChatGPT,以及副业思考、变现思路、创业案例、落地案例分享。

2、分享ChatGPT的使用方法、最新资讯、商业价值。

3、探讨未来关于ChatGPT的机遇,共同成长。

4、帮助大家解决ChatGPT遇到的问题。

5、提供一整年的售后服务,一起搞副业

星球福利:

1、加入星球4天后,就送ChatGPT独立账号。

2、邀请你加入ChatGPT会员交流群。

3、赠送一份完整的ChatGPT手册和66个ChatGPT副业赚钱手册。

其它福利还在筹划中...不过,我给你大家保证,加入星球后,收获的价值会远远大于今天加入的门票费用 !

本星球第一期原价399,目前属于试运营,早鸟价129,每超过50人涨价10元,星球马上要来一波大的涨价,如果你还在犹豫,可能最后就要以更高价格加入了。。

早就是优势。建议大家尽早以便宜的价格加入!

欢迎有需要的同学试试,如果本文对您有帮助,也请帮忙点个赞 + 在看啦!❤️

你还有什么想要补充的吗?

关键词:

Copyright   2015-2022 大西洋城建网 版权所有  备案号:沪ICP备2020036824号-2   联系邮箱: 562 66 29@qq.com