好难啊……一个 try-catch 问出这么多花样

a366bb1f211bd045f38dd8a8fc5a4e00.gif

作者 | 阿Q

来源 | 阿Q说代码

刚刚面试回来的B哥又在吐槽了:现在的面试官太难伺候了,放着好好的堆、栈、方法区不问,上来就让我从字节码角度给他分析一下try-catch-finally(以下简称TCF)的执行效率.....

今天我们就来好好总结一下~

环境准备:IntelliJ IDEA 2020.2.3、JDK 1.8.0_181

执行顺序

我们先来写一段简单的代码:

public static int test1() {int x = 1;try {return x;} finally {x = 2;}
}

答案是1不是2,你答对了吗?

大家都知道在TCF中,执行到return的时候会先去执行finally中的操作,然后才会返回来执行return,那这里为啥会是1呢?我们来反编译一下字节码文件。

命令:javap -v xxx.class

1c6323df08c2ec54963eec8adef63264.png

字节码指令晦涩难懂,那我们就用图解的方式来解释一下(我们先只看前7行指令):首先执行 int x = 1;

508d686a4867c67691a365f1ca0adcbe.png

然后我们需要执行try中的return x;

6381fc3336303a527e717e6ad16069fa.png

此时并不是真正的返回x的值,而是将x的值存到局部变量表中作为临时存储变量进行存储,也就是对该值进行保护操作。

最后进入finally中执行x=2;

8d3933241a9980de9fc09b9b5dbf126a.png

此时虽然x已经被赋值为2了,但是由于刚才的保护操作,在执行真正的return操作时,会将被保护的临时存储变量入栈返回。

为了更好的理解上述操作,我们再来写一段简单代码:

public static int test2() {int x = 1;try {return x;} finally {x = 2;return x;}
}

大家思考一下执行结果是几?答案是2不是1。

我们再来看下该程序的字节码指令

ecd774ab62766dc4883a706a8c7962eb.png

通过对比发现,第6行一个是iload_1,一个是iload_0,这是由什么决定的呢?原因就是我们上边提到的保护机制,当在finally中存在return语句时,保护机制便会失效,转而将变量的值入栈并返回。

小结

  • return的执行优先级高于finally的执行优先级,但是return语句执行完毕之后并不会马上结束函数,而是将结果保存到栈帧中的局部变量表中,然后继续执行finally块中的语句;

  • 如果finally块中包含return语句,则不会对try块中要返回的值进行保护,而是直接跳到finally语句中执行,并最后在finally语句中返回,返回值是在finally块中改变之后的值;

finally 为什么一定会执行

细心地小伙伴应该能发现,上边的字节码指令图中第4-7行和第9-12行的字节码指令是完全一致的,那么为什么会出现重复的指令呢?

首先我们来分析一下这些重复的指令都做了些什么操作,经过分析发现它们就是x = 2;return x;的字节码指令,也就是finally代码块中的代码。由此我们有理由怀疑如果上述代码中加入catch代码块,finally代码块对应的字节码指令也会再次出现。

public static int test2() {int x = 1;try {return x;} catch(Exception e) {x = 3;} finally {x = 2;return x;}
}

反编译之后

e392943a5b00c529ac60aa1ebaffb083.png

果然如我们所料,重复的字节码指令出现了三次。让我们回归到最初的问题上,为什么finally代码的字节码指令会重复出现三次呢?

原来是JVM为了保证所有异常路径和正常路径的执行流程都要执行finally中的代码,所以在trycatch后追加上了finally中的字节码指令,再加上它自己本身的指令,正好三次。这也就是为什么finally 一定会执行的原因。

finally一定会执行吗?

为什么上边已经说了finally中的代码一定会执行,现在还要再多此一举呢?请👇看

在正常情况下,它是一定会被执行的,但是至少存在以下三种情况,是一定不执行的:

  • try语句没有被执行到就返回了,这样finally语句就不会执行,这也说明了finally语句被执行的必要而非充分条件是:相应的try语句一定被执行到;

  • try代码块中有System.exit(0);这样的语句,因为System.exit(0);是终止JVM的,连JVM都停止了,finally肯定不会被执行了;

  • 守护线程会随着所有非守护线程的退出而退出,当守护线程内部的finally的代码还未被执行到,非守护线程终结或退出时,finally 肯定不会被执行;

TCF 的效率问题

说起TCF的效率问题,我们不得不介绍一下异常表,拿上边的程序来说,反编译class文件后的异常表信息如下:

4f58ef654e49074844c46b1b5bcb4778.png
  • from:代表异常处理器所监控范围的起始位置;

  • to:代表异常处理器所监控范围的结束位置(该行不被包括在监控范围内,是前闭后开区间);

  • target:指向异常处理器的起始位置;

  • type:代表异常处理器所捕获的异常类型;

图中每一行代表一个异常处理器

工作流程:

  1. 触发异常时,JVM会从上到下遍历异常表中所有的条目;

  2. 比较触发异常的行数是否在from-to范围内;

  3. 范围匹配之后,会继续比较抛出的异常类型和异常处理器所捕获的异常类型type是否相同;

  4. 如果类型相同,会跳转到target所指向的行数开始执行;

  5. 如果类型不同,会弹出当前方法对应的java栈帧,并对调用者重复操作;

  6. 最坏的情况下JVM需要遍历该线程 Java 栈上所有方法的异常表;

拿第一行为例:如果位于2-4行之间的命令(即try块中的代码)抛出了Class java/lang/Exception类型的异常,则跳转到第8行开始执行。

8: astore_1是指将抛出的异常对象保存到局部变量表中的1位置处

从字节码指令的角度来讲,如果代码中没有异常抛出,TCF的执行时间可以忽略不计;如果代码执行过程中出现了上文中的第6条,那么随着异常表的遍历,更多的异常实例被构建出来,异常所需要的栈轨迹也在生成。该操作会逐一访问当前线程的栈帧,记录各种调试信息,包括类名、方法名、触发异常的代码行数等等。所以执行效率会大大降低。

看到这儿,你是否对TCF有了更加深入的了解呢?

078ab2515d86b9559665a15b68d14fb2.gif

往期推荐

为什么还有这么多的网络故障?

k8s集群居然可以图形化安装了?

用了HTTPS,没想到还是被监控了

将 k8s 制作成 3D 射击游戏,好玩到停不下来

fe712012664dc09136f07fe05b2b21ec.gif

点分享

08db71eebc3bfe69157efac680ec710f.gif

点收藏

af3173d784145fb31a7b623e73eb3040.gif

点点赞

a1b841eabdbf2d5901bc44c82036336d.gif

点在看

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/512433.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据是如何被保护的?高质量存储告诉你

简介: 作为关键信息基础设施运营者,阿里云提供了全方位的数据安全保护方案。今天,我们就从数据存储的角度来聊一聊数据是如何被保护的。 原文链接 本文为阿里云原创内容,未经允许不得转载。

无法启动此程序因为计算机丢失dtlui,电脑缺少dll文件_电脑开机总是出来DLL文件丢失,...

最佳答案如果您的电脑开机出现“加载C:/WIND/SYSTEM32.UKWIEG96.DLL时出错”之类的加载dll文件出错或提示dll文件丢失,但又可以正常进入系统,那么这篇文章或许可以给您带来解决办法。碰到这种情况我们可以肯定是dll因为某些原因出错或丢失了&…

findler mac 隐藏文件_fiddler使用实例之----------查找隐藏的真实地址!!!!

这个教程讲解下如何用fddler ,找到页面的真是地址同样也是可以找到页面隐藏的内容,两者原理是一样的,在页面能看到的信息,页面源代码却看不到,也就是这样的信息其实并不在这个页面上,而是通过一些方法调用来…

Nacos 开源、自研、商业化三位一体战略解读

简介: Nacos作为整个阿里云原生三位战略中的核心组成部分,我们在2018年以Configserver/VIPServer/Diamond为基础通过Nacos开源输出阿里十年沉淀的注册中心和配置中心能力,并且快速成为国内首选。并且通过云产品MSE以BaaS模式输出解决方案能力…

Haystack 太强了!存 2600 亿图片

作者 | 奇伢来源 | 奇伢云存储小文件存储小文件存储,老生常谈的问题。先聊聊小文件存储重点关注的是什么?以前我们提过,对于磁盘来说,小 io 吃 iops,大块 io 吃吞吐。划重点:小文件的重点是 io 次数。为什么…

全国计算机网络教学研讨会,历届全国高校计算机网络教学研讨会

1. 第六届全国高校计算机网络教学研讨会在内蒙古大学成功召开2. 第五届全国高校计算机网络教学研讨会在温州大学成功召开 2012年10月26日-27日,第五届全国高校计算机网络教学研讨会在温州大学召开。会议由中国计算机学会互联网专业委员会和中国计算机学会…

Dubbo 和 HSF 在阿里巴巴的实践:携手走向下一代云原生微服务

简介: HSF 和 Dubbo 的融合是大势所趋。为了能更好的服务内外用户,也为了两个框架更好发展,Dubbo 3.0 和以 Dubbo 3.0 为内核适配集团内基础架构生态的 HSF 3 应运而生。 作者 | 郭浩 Dubbo 和 HSF 都是阿里巴巴目前在使用的微服务 RPC 框架…

apache过滤恶意频繁访问_采用网关过滤器实现权限验证及对异常统一处理

采用网关过滤器实现权限验证1、创建 zuul 项目2、修改 pom.xml 文件<project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.apache.org/POM/4.0.0 http://mav…

如何攻破容器持久化存储挑战?

背景 云原生趋势下&#xff0c;应用容器化比例正在快速增长&#xff0c;Kubernetes 也已成为云原生时代新的基础设施。 观察今天的容器和 Kubernetes 的应用现状&#xff0c;可以看到两个普遍的现象&#xff1a; 首先&#xff0c;在云上托管 Kubernetes 已经成为企业上云及运…

用科技共创美好:英特尔助力北京冬奥会新体验

2022年1月24日&#xff0c;北京2022年冬奥会在即&#xff0c;北京冬奥会的气氛愈发浓烈。作为奥运会全球TOP合作伙伴&#xff0c;英特尔以基于英特尔处理器的创新技术支持奥运会&#xff0c;释放科技冬奥的魅力。今天&#xff0c;英特尔在位于前门建成的全新升级的“英特尔北京…

职称计算机Word2003是考什么,2017年职称计算机考试word2003考点

2017年职称计算机考试word2003考点计算机在我们的工作中太重要了&#xff0c;很多工作岗位对计算机都有一定的要求。以下是小编整理的2017年职称计算机考试word2003考点&#xff0c;希望可以为您的学习带来帮助!内置段落样式1、套用段落样式&#xff1a;选中要套用样式的一个或…

李飞飞:新技术变革时代的数据库产业

简介&#xff1a; 云计算将改变数据库格局 近日&#xff0c;阿里云智能数据库事业部负责人李飞飞在媒体沟通会上发表了“新技术变革时代的数据库产业”主题演讲。 李飞飞说&#xff0c;云数据库已经成为数据库最重要的发展方向&#xff0c;从国际国内数据库产业的发展来看&am…

iostat命令详解_对iostat输出结果的理解

前言&#xff1a;日常工作中&#xff0c;线上服务会出现各种奇奇怪怪的问题&#xff0c;每次出现问题都是根据现象猜测出现问题的原因&#xff0c;比如请求响应慢了&#xff0c;就排查整个请求的逻辑&#xff0c;每一步花了多少时间&#xff0c;分析半天终于发现是某一步慢了以…

计算机windows10属性配置,电脑显示属性设置,教你win10系统电脑显示属性的设置教程...

今天小编教你win10系统电脑显示属性的设置教程&#xff0c;显卡作为计算机最为基本的配置和最重要的配件之一&#xff0c;承担着输出显示图形的任务。不知电脑显卡设置在哪里打开及如何设置的用户&#xff0c;请来看看下面的介绍吧。显卡是一台电脑的第二个核心&#xff0c;我们…

庖丁解牛-图解MySQL 8.0优化器查询解析篇

简介&#xff1a; SQL优化器本质上是一种高度抽象化的数据接口的实现&#xff0c;经过该设计&#xff0c;客户可以使用更通用且易于理解的SQL语言&#xff0c;对数据进行操作和处理&#xff0c;而不需要关注和抽象自己的数据接口&#xff0c;极大地解放了客户的应用程序。 作者…

历时 4 年,阿里云推出金融核心系统转型实践书

在过去 40 年中&#xff0c;基本由国外技术支撑着我国金融行业核心系统的建设。随着近十年来&#xff0c;中国互联网飞速发展&#xff0c;在数字金融领域&#xff0c;中国正发生巨变&#xff0c;如手机支付、纳税平台应用、风控系统等业务的普及&#xff0c;以往国外的实践经验…

三款典型国产分布式数据库的对比评测

简介&#xff1a; 编者按&#xff1a;近几年国产数据库市场风生水起&#xff0c;涌现了多款优秀的国产数据库产品&#xff0c;本文选取了三款典型的国产分布式数据库进行全方位对比压测&#xff0c;呈现了国产分布式数据库的发展现状。 对于所有的应用系统&#xff0c;数据都是…

bootstraptable中responsehandle获取数据缺失_Python中的向量化字符串操作

Python的一个使用优势是它在处理和操作字符串数据方面相对容易。在此基础上Pandas提供了一套全面的向量化字符串操作(vectorized string operation)&#xff0c;这些操作成为处理现实世界数据时所需的必不可少的功能。在本文中&#xff0c;我们将介绍一些Pandas的字符串操作&am…

计算机二级的编程题,计算机二级编程题(范文).doc

第 PAGE \* Arabic 1 页计算机二级编程题(范文)整数排序题☆题目1(整数排序题)在文件in.dat中有200个正整数&#xff0c;且每个数均在1000至9999之间。函数ReadDat()读取这200个数存放到数组aa中。请编制函数jsSort()&#xff0c;其函数的功能是&#xff1a;要求按每个数的后三…

15 载专注视频增强技术,小而美的 Imint 蕴藏大匠心

如今视频已深深融入我们的生活和工作中&#xff0c;据 CNNIC 数据显示&#xff0c;截止 2021 年 6 月&#xff0c;我国网民的规模达 10.11 亿&#xff0c;其中短视频用户规模 8.88 亿&#xff0c;占网民整体的 87.8%。 这表明我们正步入“视频社会化”时代&#xff0c;随着人们…