PointerException 的处理新方式,Java14 真的太香了[通俗易懂]

PointerException 的处理新方式,Java14 真的太香了[通俗易懂]本质上,JEP 358 旨在通过描述某个变量是 “”来提高 JVM 生成的 “PointerException” 的可读性。

大家好,欢迎来到IT知识分享网。

优质文章,及时送达

来源:锅外的大佬

Java语言中,处理空指针往往是一件很头疼的事情,一不小心,说不定就搞出个线上Bug,让你的绩效考核拿到3.25。

最近新出的Java14,相信大家都有所耳闻,那么今天就来看看,面对PointerException,Java14有哪些更好的处理方式呢?

1.传统的 PointerException

我们编码过程中呢,经常会使用链式调用的方式来写代码,这样写起来很方便,也很清晰,但是,一旦出现PointerException,那就头大了,因为你很难知道异常是在什么时候开始发生的。

举个简单的例子,就比如下面的代码,要找到公司某个员工的户籍所在地,我们这样来调用

String city = employee.getDetailInfos.getRegistryAddress.getCity;

IT知识分享网

在链式调用的过程中,如果employee,getDetailInfos,或者getRegistryAddress为空,JVM就会抛出PointerException

那么导致异常的根本原因是什么?如果不使用调试器,很难确定哪个变量为空。而且,JVM也只会打印导致异常的方法、文件名和行号,仅此而已。那么下面,我将带大家了解Java 14如何通过 JEP 358解决这个问题。

2.增强型 PointerException

SAP在2006年为其商业JVM实现了增强型的 PointerException。2019年2月,它被提议作为OpenJDK社区的一个增强,之后很快,它成为了一个JEP。所以,该功能在2019年10月完成并在JDK 14版本推出

本质上,JEP 358 旨在通过描述某个变量是 “来提高 JVM 生成的 “PointerException” 的可读性。JEP 358通过在方法、文件名和行号旁边描述为 的变量,带来了一个详细的PointerException消息。它通过分析程序的字节码指令来工作。因此,它能够精确地确定哪个变量或表达式是。最重要的是,JDK 14中默认关闭详细的异常消息。要启用它,我们需要使用命令行选项:

IT知识分享网-XX:+ShowCodeDetailsInExceptionMessages

2.1 详细的异常信息

考虑在激活 ShowCodeDetailsInExceptionMessages标志的情况下再次运行代码:

Exception in thread "main" java.lang.PointerException:
Cannot invoke "RegistryAddress.getCity" because the return value of
"com.developlee.java14.helpfulpointerexceptions.HelpfulPointerException$DetailInfos.getRegistryAddress" is
at com.developlee.java14.helpfulpointerexceptions.HelpfulPointerException.main(HelpfulPointerException.java:10)

这一次,从附加信息中,我们知道员工的个人详细信息丢失的注册地址导致了我们的异常。从这个增强中获得的信息可以节省我们调试所用的时间。

JVM由两部分组成详细的异常消息。第一部分表示失败的操作,这是引用为 ** 的结果,而第二部分标识了 ** 引用的原因:

IT知识分享网Cannot invoke "String.toLowerCase" because the return value of "getEmailAddress" is 

为了生成异常消息,JEP 358 重构了将空引用推送到操作数堆栈上的部分源代码。

3. 技术方面

现在我们已经很好地理解了如何使用增强的PointerExceptions标识 引用,让我们来看看它的一些技术方面。

首先,只有当JVM本身抛出一个 PointerException时,才会进行详细的消息计算,如果我们在Java代码中显式抛出异常,则不会执行计算。原因是因为:在这些情况下,很可能已经在异常构造函数中传递了一条有意义的消息。

其次,**JEP 358 ** 懒汉式地计算消息,这意味着只有当我们打印异常消息时才调用增强的PointerException,而不是当异常发生时就调用。因此,对于通常的JVM流程不应该有任何性能影响,在那里我们可以捕获并重新抛出异常,因为咱并不会只想打印异常消息。

最后,详细的异常消息可能包含源代码中的局部变量名。因此,我们可以认为这是一个潜在的安全风险。但是,只有在运行使用激活的-g标记编译的代码时,才会发生这种情况,该标记会生成调试信息并将其添加到类文件中。请考虑一个简单的示例,我们已编译该示例以包含以下附加调试信息:

Employee employee = ;
employee.getName;

当执行以上代码时,异常信息中会打印本地变量名称:

"com.developlee.java14.helpfulpointerexceptions.HelpfulPointerException$Employee.getName"
because "employee" is

相反,在没有额外调试信息的情况下,JVM 只提供它在详细消息中所知道的变量:

Cannot invoke
"com.developlee.java14.helpfulpointerexceptions.HelpfulPointerException$Employee.getName"
because "<local1>" is

JVM 打印编译器分配的变量索引,而不是本地变量名(employee)。

关于PointerException的处理到这里就结束了,通过Java14增强的PointerException,我们可以很快速的定位代码问题的原因所在,更快的调试代码,节约时间,提高效率。

已经安装了Java14的朋友可以试试看哦。

-END-

如果看到这里,说明你喜欢这篇文章,请 。同时 标星(置顶)本公众号可以第一时间接受到博文推送。

PointerException 的处理新方式,Java14 真的太香了[通俗易懂]

最近整理一份面试资料《Java技术栈学习手册》,覆盖了Java技术、面试题精选、Spring全家桶、Nginx、SSM、微服务、数据库、数据结构、架构等等。

免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://yundeesoft.com/13446.html

(0)
上一篇 2023-03-08 17:30
下一篇 2023-03-08 18:30

相关推荐

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注微信