游戏电视苹果数码历史美丽
投稿投诉
美丽时装
彩妆资讯
历史明星
乐活安卓
数码常识
驾车健康
苹果问答
网络发型
电视车载
室内电影
游戏科学
音乐整形

告别空指针让代码变优雅,Optional使用图文例子源码解读

  一、前言
  我们在开发中最常见的异常就是NullPointerException,防不胜防啊,相信大家肯定被坑过!
  这种基本出现在获取数据库信息中、三方接口,获取的对象为空,再去get出现!
  解决方案当然简单,只需要判断一下,不是空在去后续操作,为空返回!
  所有在JDK8时出现了专门处理的方案,出来很早了,但是小编惭愧一直没有去使用它!
  最近在看《Java开发手册》,一直想着提高自己的代码水平,文中就指出了使用Optional来解决NullPointerException!二、Java开发手册规范
  小编使用的是2022版的黄山版,29页写到:
  【推荐】防止NPE,是程序员的基本修养,注意NPE产生的场景:返回类型为基本数据类型,return包装数据类型的对象时,自动拆箱有可能产生NPE
  反例:publicintmethod(){returnInteger对象;},如果为null,自动解箱抛NPE。数据库的查询结果可能为null。集合里的元素即使isNotEmpty,取出的数据元素也可能为null。远程调用返回对象时,一律要求进行空指针判断,防止NPE。对于Session中获取的数据,建议进行NPE检查,避免空指针。级联调用obj。getA()。getB()。getC();一连串调用,易产生NPE。
  正例:使用JDK8的Optional类来防止NPE问题。
  这份手册还是不错的,推荐反复阅读,虽然进不去大厂,也要自觉约束自己的代码风格,努力向大厂靠!
  大家现在不知道哪里找的可以下载一下:
  《Java开发手册》:https:github。comalibabap3c三、Optional常用方法
  小编带大家一起从api文档中的方法,一个个带大家慢慢去了解它!1。empty()
  返回一个空的Optional实例:Optional。emptyOptionalObjectemptyOptional。empty();log。info(empty值:{},empty);
  在这里插入图片描述2。of(Tvalue)
  传入一个参数,返回一个Optional对象,如果参数为空,报NullPointerException!TesttestNewnewTest();Testtestnull;OptionalTestoptionalNewOptional。of(testNew);log。info(optional对象:{},optionalNew);OptionalTestoptionalOptional。of(test);
  在这里插入图片描述
  源码查看:
  我们看到参数为空会报NullPointerException,我们去方法内部看一下就明白了:publicstaticTOptionalTof(Tvalue){returnnewOptional(value);}privateOptional(Tvalue){this。valueObjects。requireNonNull(value);}publicstaticTTrequireNonNull(Tobj){if(objnull)thrownewNullPointerException();returnobj;}
  我们发现是在Objects类中的requireNonNull方法中判断了是否为空!
  这个还会出现NullPointerException,所以我们一般使用下面的这个方法!3。ofNullable(Tvalue)
  参数传入一个对象,返回一个Optional对象,如果为空,将返回一个空的Optional对象,就等于Optional。emptyTesttestNewnewTest();Testtestnull;OptionalTestoptionalNewOptional。of(testNew);log。info(optional对象:{},optionalNew);OptionalTestoptionalTestOptional。ofNullable(test);log。info(optional对象中的ofNullable方法返回值:{},optionalTest);OptionalTestoptionalTestNewOptional。ofNullable(testNew);log。info(optional对象中的ofNullable方法new返回值:{},optionalTestNew);
  在这里插入图片描述
  源码查看:
  我们发现是在方法开始进行非空判断,再去调用上面的of(Tvalue)方法publicstaticTOptionalTofNullable(Tvalue){returnvaluenull?empty():of(value);}4。get()
  如果此Optional中存在值,则返回该值,否则抛出NoSuchElementException。TesttestNewnewTest();Testtestnull;OptionalTestoptionalNewOptional。of(testNew);log。info(optional对象:{},optionalNew);OptionalTestoptionalOptional。of(test);OptionalTestoptionalTestOptional。ofNullable(test);log。info(optional对象中的ofNullable方法返回值:{},optionalTest);OptionalTestoptionalTestNewOptional。ofNullable(testNew);log。info(optional对象中的ofNullable方法new返回值:{},optionalTestNew);Testtest2optionalTestNew。get();log。info(原来有值的:经过Optional包装后get后得到原来的值:{},test2);Testtest1optionalTest。get();log。info(原来没有值的:经过Optional包装后get后得到原来的值:{},test1);
  在这里插入图片描述
  源码查看:
  调用开始会进行值判断,如果为空则抛异常!publicTget(){if(valuenull){thrownewNoSuchElementException(Novaluepresent);}returnvalue;}5。isPresent()
  如果存在值,则返回true,否则返回false。
  这里代码就不加上面的,大家参考上面的获取一个Optional对象booleanpresentoptionalTestNew。isPresent();log。info(optionalTestNew调用是否为空:{},present);booleanpresent1optionalTest。isPresent();log。info(optionalTest调用是否为空:{},present1);
  在这里插入图片描述
  源码查看:
  这就比较简单了!publicbooleanisPresent(){returnvalue!null;}6。ifPresent(Consumerconsumer)
  如果存在值,则使用该值调用指定的使用者,否则不执行任何操作。
  主要的就是入参数一个函数式接口,有值就会去执行,为空则不进行任何操作!
  小技巧:
  开始对lambda不了解时,可以先按照上面这种方式进行写,
  大家可以看到Idea给置灰了,就是可以优化,我们AltEnter
  然后再次Enter就会变成后面的lambda!
  在这里插入图片描述
  optionalTest。ifPresent(newConsumer(){
  Override
  publicvoidaccept(Testtest){
  log。info(我是调用ifPresent执行后的打印);
  }
  });
  optionalTestNew。ifPresent(testInnerlog。info(我是调用ifPresent执行后的打印));
  在这里插入图片描述
  源码查看:
  还是先判断不为空才去执行函数式接口!publicvoidifPresent(Consumerlt;?superTconsumer){if(value!null)consumer。accept(value);}7。filter(Predicatepredicate)
  如果存在值,并且该值符合规则,则返回描述该值的Optional,否则返回空Optional
  是一个Predicate函数接口,可以传入实现了Predicate接口的lambda表达式!
  如果不符合条件就会返回一个Optional。emptytestNew。setName(萧炎);testNew。setAge(33);OptionalTestoptionalTest1optionalTestNew。filter(test1test1。getAge()30);log。info(过滤后的结果:{},optionalTest1。get());
  在这里插入图片描述
  源码查看:
  就是判断一下表达式和值是否为空,然后就是根据规则判断publicOptionalTfilter(Predicatelt;?superTpredicate){Objects。requireNonNull(predicate);if(!isPresent())returnthis;elsereturnpredicate。test(value)?this:empty();}8。map(Functionmapper)
  如果存在值,则将提供的映射函数应用于该值,如果结果为非空,则返回描述结果的Optional。否则,返回空的Optional。
  也是一个函数式接口!OptionalStringstringOptionaloptionalTestNew。map(Test::getName);log。info(map后获得字段值:{},stringOptional。get());
  在这里插入图片描述
  源码查看:
  也是进行非空判断,然后执行lambda得到字段后放到ofNullable方法中!publicUOptionalUmap(Functionlt;?superT,?extendsUmapper){Objects。requireNonNull(mapper);if(!isPresent())returnempty();else{returnOptional。ofNullable(mapper。apply(value));}}9。flatMap(Functionmapper)
  如果存在值,则将提供的Optional方位映射函数应用于该值,返回该结果,否则返回空的Optional。此方法类似于map,但提供的映射器的结果已经是可选的,并且如果调用,flatMap不会不会在最后进行任何包装。OptionalStringoptionaloptionalTestNew。flatMap(OptionalTest::getFlatMap);log。info(flatMap后得到的字段:{},optional。get());privatestaticOptionalStringgetFlatMap(Testtest){returnOptional。ofNullable(test)。map(Test::getName);}
  在这里插入图片描述
  源码查看:
  也是进行非空判断,然后和map不同的是不执行ofNullable方法publicUOptionalUflatMap(Functionlt;?superT,OptionalUmapper){Objects。requireNonNull(mapper);if(!isPresent())returnempty();else{returnObjects。requireNonNull(mapper。apply(value));}}10。orElse(Tother)
  如果有值则将其返回,否则返回指定的其它值。
  如果你是一个对象,orElse()也要是相同对象!Stringmessagenull;StringmessageNew关注公众号:小王博客基地;StringnullStringOptional。ofNullable(message)。orElse(这是一个空字符串!);log。info(这是空字符串打印的:{},nullString);StringstringOptional。ofNullable(messageNew)。orElse(这是一个空字符串!);log。info(这是字符串打印的:{},string);
  在这里插入图片描述
  源码查看:
  简单的为空返回自己定义的,不为空直接返回!publicTorElse(Tother){returnvalue!null?value:other;}11。orElseGet(Supplierother)
  返回值(如果存在),否则调用other并返回该调用的结果。
  区别:
  orElse方法将传入的参数作为默认值,orElseGet方法可以接受Supplier接口的实现用来生成默认值
  如果没有复杂操作,Idea也会提醒我们不要使用这个,使用orElse即可!Stringmessagenull;StringmessageNew关注公众号:小王博客基地;StringorElseGetOptional。ofNullable(message)。orElseGet(()这还是一个空的字符串);log。info(orElseGet调用:这是空字符串打印的:{},orElseGet);StringorElseGetStringOptional。ofNullable(messageNew)。orElseGet(()这还是一个空的字符串);log。info(orElseGet调用:这是字符串打印的:{},orElseGetString);
  在这里插入图片描述
  源码查看:
  和orElse一样,只不过为空调用lambda执行!publicTorElseGet(Supplierlt;?extendsTother){returnvalue!null?value:other。get();}12。orElseThrow(SupplierexceptionSupplier)
  返回包含的值(如果存在),否则抛出由提供的供应商创建的异常。Stringmessagenull;StringmessageNew关注公众号:小王博客基地;Optional。ofNullable(messageNew)。orElseThrow(()newRuntimeException(为空了,还不看看!));Optional。ofNullable(message)。orElseThrow(()newRuntimeException(为空了,还不看看!));
  我们可以自定义异常,然后来引用!
  在这里插入图片描述
  源码查看:
  为空则走自己写的异常!publicXextendsThrowableTorElseThrow(Supplierlt;?extendsXexceptionSupplier)throwsX{if(value!null){returnvalue;}else{throwexceptionSupplier。get();}}13。例子汇总authorwangzhenjundate202322710:22Slf4jpublicclassOptionalTest{publicstaticvoidmain(String〔〕args){OptionalObjectemptyOptional。empty();log。info(empty值:{},empty);TesttestNewnewTest();Testtestnull;OptionalTestoptionalNewOptional。of(testNew);log。info(optional对象:{},optionalNew);OptionalTestoptionalOptional。of(test);OptionalTestoptionalTestOptional。ofNullable(test);log。info(optional对象中的ofNullable方法返回值:{},optionalTest);OptionalTestoptionalTestNewOptional。ofNullable(testNew);log。info(optional对象中的ofNullable方法new返回值:{},optionalTestNew);Testtest2optionalTestNew。get();log。info(原来有值的:经过Optional包装后get后得到原来的值:{},test2);Testtest1optionalTest。get();log。info(原来没有值的:经过Optional包装后get后得到原来的值:{},test1);booleanpresentoptionalTestNew。isPresent();log。info(optionalTestNew调用是否为空:{},present);booleanpresent1optionalTest。isPresent();log。info(optionalTest调用是否为空:{},present1);optionalTest。ifPresent(newConsumerTest(){Overridepublicvoidaccept(Testtest){log。info(我是调用ifPresent执行后的打印);}});optionalTestNew。ifPresent(testInnerlog。info(我是调用ifPresent执行后的打印));testNew。setName(萧炎);testNew。setAge(33);OptionalTestoptionalTest1optionalTestNew。filter(test1test1。getAge()30);log。info(过滤后的结果:{},optionalTest1。get());OptionalStringstringOptionaloptionalTestNew。map(Test::getName);log。info(map后获得字段值:{},stringOptional。get());OptionalStringoptionaloptionalTestNew。flatMap(OptionalTest::getFlatMap);log。info(flatMap后得到的字段:{},optional。get());Stringmessagenull;StringmessageNew关注公众号:小王博客基地;StringnullStringOptional。ofNullable(message)。orElse(这是一个空字符串!);log。info(这是空字符串打印的:{},nullString);StringstringOptional。ofNullable(messageNew)。orElse(这是一个空字符串!);log。info(这是字符串打印的:{},string);StringorElseGetOptional。ofNullable(message)。orElseGet(()这还是一个空的字符串);log。info(orElseGet调用:这是空字符串打印的:{},orElseGet);StringorElseGetStringOptional。ofNullable(messageNew)。orElseGet(()这还是一个空的字符串);log。info(orElseGet调用:这是字符串打印的:{},orElseGetString);Optional。ofNullable(messageNew)。orElseThrow(()newRuntimeException(为空了,还不看看!));Optional。ofNullable(message)。orElseThrow(()newRuntimeException(为空了,还不看看!));}privatestaticOptionalStringgetFlatMap(Testtest){returnOptional。ofNullable(test)。map(Test::getName);}}四、总结
  这里就不在演示实战了,基本上组合使用:
  Optional。ofNullable(需要判断的对象)。ifPresent(具体操作)
  其实和if相比就是显得优雅一些,主要是防止某处没考虑到,忘记if判断,那么后续可能会导致空指针,如果使用Optional的话,那么这个问题能够得到避免。
  就像多使用设计模式一样,让自己的代码更加健壮优雅,还是要多使用一些的!当然不能过渡使用!!
  对你有帮助,还请不要吝啬你的发财小手点点关注哈!、
  写作不易,大家给点支持,你的支持是我写作的动力哈!
  关注小编的微信公众号,一起交流学习!文章首发看哦!

手游生存几何将于15日上线《生存几何》是一款末日题材的模拟经营类游戏,在这里,玩家将扮演一名在末日城市发展的新老板,为了生存,撸起袖子开始工作吧!让你的商店在城市中名声大噪,大赚特赚!自定义装修商……60岁的叶倩文状态好,无子分房睡,也不影响她与林子祥的幸福最近芒果台出了一档《声生不息》的综艺,一上线就收获了超高的人气和口碑,看样子,这档综艺要成为芒果台今年的爆款了。节目组很有心,同时请来了林子祥和叶倩文这对已经白发苍苍的夫……把奢侈大牌穿成地摊货,这些明星简直是时尚灾难啊俗话说人靠衣装,而作为娱乐圈中星光熠熠的明星们,动辄出席活动都有各种造型师,私底下也有不少明星是穿搭达人。但也有一些明星简直是时尚灾难,不管多时髦的奢侈大牌都能穿出地摊货的廉价……收下这份高端家电产品清单,从此开启快乐生活的新征程现代生活的快节奏感,让每个人的步履都变得匆匆起来。对于大多数上班族来说,每天两点一线的生活,看见的不是公司就是家。家就更像是我们忙碌一天后,可以卸下疲惫,获取能量的充电堡……马可罗潮玩牛仔值不值得买?技能对比暗影游猎后,60点券真香王者荣耀中射手英雄有很多,但是热门的英雄数量相对来说比较少,目前一百多位英雄人物,经常出场的自然是比较少的,所以官方才会经常对一些英雄进行加强或者削弱来调整游戏里面的公平性,马……你们都是托买了SSD就能10秒开机?虽然很多玩家都不知道SSD的工作方式,但这并不妨碍大家知道SSD能带来极速体验,尤其是10秒开机,让不少玩家心驰神往。但是买了SSD就能10秒开机吗?还真的未必。为什么呢……2022都有哪些古装剧值得期待?2022已经开启,1月份就上了不少大剧和好剧,但是基本都是生活剧、悬疑剧,古装剧暂时还没有爆发,数量零星,话题也只有《镜双城》稍微高点,口碑还不算优秀,目测是不会大爆了。……AG又夺冠了,CF选手赢麻了,老板菲菲发文祝贺,AG超玩会却CF这款游戏可以说承载了很多人的青春回忆,在电竞比赛如此火爆的时代,CF赛事也备受大家的喜爱,现在更拥有很多优秀的电竞赛事。在前不久刚结束的世界赛中,我们也有很多强悍的队伍进入……当拳头开始做FPS电竞VALORANT的第一届S赛在马拉松一般艰难的五场比赛结束后,来自欧洲的俱乐部Acend赢得了今年VALORANT全球冠军赛的最终胜利,成为了这个新FPS项目的第一个世界冠军。这场上周刚刚结束的拳头……在海外打过球,被朱芳雨看中,加入广东队!他现在怎样了?广东队在CBA第一阶段比赛中,取得11胜2负的战绩,高居联盟第2位,仅次于辽宁队。这赛季阿联迎来复出,但在第一阶段的比赛中,阿联出场次数和场均上场时间并不多,他出战了6场……美国应该只剩下一条路衰退美国应该只剩下一条路:衰退!且这一次衰退不亚于1929年的经济大萧条。美国印钱太多,资产规模大到吓人,砸下来山崩地裂!由于钱太多,人们的收入增加超过20,导致消费旺……CBA场上又起冲突,球员互欧,篮球场上一片混乱2022年1月13日晚中国篮球CBA常规赛中,广东宏远队对阵辽宁篮球队,辽宁队韩德君与广东队外援威姆斯因场上身体冲撞而大打出手,裁判将两人双双驱逐出场。CBA联盟给予两人严厉处……
车前草能治疗痛风吗?它是对付痛风的宝平车前,学名PlantagodepressaWilld。,俗称车前草。一年生或二年生草本。直根长,具多数侧根,多少肉质。根茎短。下面5号网小编带大家来看一下车前草能治疗痛风吗?……喝酒会引起痛风吗?喝酒会引起面瘫吗?喝酒会给我们的身体带来很多坏处,尤其是对肝脏,影响非常大,另外还有很多其他的害处,所以大家千万不要酗酒哦。下面5号网小编给大家讲讲喝酒会引起痛风吗?喝酒会引起面瘫吗?喝酒会引起……为什么会白带异常呢是什么原因造成白带异常呢大家在生活中不知道遇见过白带异常的情况吗?今天小编就和大家一起来了解一下吧,究竟为什么会白带异常呢,以及是什么原因造成白带异常呢,跟着小编我们一起来了解吧。为什么会白带异常呢……白带异常要如何预防6种方法教你解决白带异常说明女性有妇科疾病,那么,白带异常要如何预防呢?预防白带异常的方法1、定期检查即便无任何不适也应该定期检查,最好每年至少做一次全面的妇科体检。2、及时……阿莫西林能随便吃吗?乱服阿莫西林有害无益阿莫西林被人们认为是很安全的药品,那么这款药是否能随意吃呢,下面5号网的小编为你们介绍阿莫西林能随便吃吗?乱服阿莫西林有害无益。阿莫西林能随便吃吗不要随便吃这些抗生素类的……阿莫西林能治肾炎吗?阿莫西林对肾炎有一定作用阿莫西林是非常好用的消炎药,很多人都吃过的,下面5号网的小编为你们介绍阿莫西林能治肾炎吗?阿莫西林对肾炎有一定作用。阿莫西林能治肾炎吗主要是抗生素消炎作用,对于泌尿系炎症……阿莫西林不良反应是什么?阿莫西林不良反应约为百分之五阿莫西林虽然比较出名,但是相信很多人都不知道阿莫西林的不良反应,下面5号网的小编为你们介绍阿莫西林不良反应是什么?阿莫西林不良反应约为百分之五。阿莫西林不良反应是什么临床……阿莫西林好还是阿奇霉素好?消炎药哪家强阿莫西林是比较出名的消炎药了,那么消炎药肿中到底哪种最好呢,下面5号网的小编为你们介绍阿莫西林好还是阿奇霉素好?消炎药哪家强。阿莫西林好还是阿奇霉素好阿莫西林杀菌面广,阿……Lazada速卖通美客多国际站测评自养号如何提高店铺排名权重在lazada在平台上开店后,商家需要更加关注商店的权重。当商家的商店的权重越高,流量和订单就越多,所以在lazada、速卖通、阿里国际平台上开店应该如何提高权重?以下是一些提……华为6000mAh巨能量6。75英寸大屏鸿蒙OS,亲民价10时代在发展,人们的生活水平在不断的提高,电子设备在平时已经是很常见的产品了。其中手机是市场保有量最大的,在国内几乎是人手一部了。这些年的手机更新迭代非常快,大家为了顺应时代的发……癫痫应该注意什么?癫痫属于常见病,发病率高,对患者的身心会造成影响,患者要积极治疗,也要注意饮食,多吃些营养丰富的食物,新鲜的蔬菜水果都需要多吃些,还要多补充矿物质,但是要少吃些含钾和含锌元素多……美国宇航局拍摄到太阳向我们ampampquot微笑ampam美国宇航局的太阳动力学观测站(SDO)上周拍摄了一张太阳的肖像,使它看起来有两只黑眼睛,一个旋转的圆鼻子和并向我们微笑。说茄子!美国宇航局在Twitter上发布了这张引人注目的……
友情链接:易事利快生活快传网聚热点七猫云快好知快百科中准网快好找文好找中准网快软网