好难啊……一个 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,一经查实,立即删除!

相关文章

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

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

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;极大地解放了客户的应用程序。 作者…

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

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

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

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

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

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

Serverless Devs 2.0 全新发布,让 Serverless 应用开发更简单

简介&#xff1a; 2020 年 10 月 23日&#xff0c;阿里巴巴正式宣布开源其首个 Serverless 开发者平台 Serverless Devs。历经近一年精心打磨&#xff0c;今天 Serverless Devs 2.0 正式版全新发布。Serverless Devs 2.0 在平台能力、应用模板以及开发者套件方面能力提升&#…

疫情防控“漫入调查系统”上线 SENSORO 助力提升筛查效率及精准度

连日来&#xff0c;国内多地报告新增病例&#xff0c;加上因春节临近导致的人员流动和聚集增加&#xff0c;基层防疫面临着比平时更大的挑战。为快速、高效地解决大规模漫入信息筛查任务&#xff0c;缓解一线疫情防控压力&#xff0c;SENSORO&#xff08;北京升哲科技有限公司&…

程序媛如何自我突破?

简介&#xff1a; 很多时候人们是被自己内心的偏见所打败的。作为一名程序媛&#xff0c;保持一种对世界、对人生的不同看法&#xff0c;可以帮助我们树立自己的参照系&#xff0c;不被外部轻易左右。或许我们无法像一些伟人那样打破、推动如此重大的社会认知&#xff0c;但是我…

如何基于Dataphin实现敏感数据保护

简介&#xff1a; 在企业的发展过程中&#xff0c;如果不重视敏感数据的保护&#xff0c;和数据安全体系的建设&#xff0c;那么一旦发生了敏感数据泄漏事件&#xff0c;轻则企业口碑受损&#xff0c;业务受影响&#xff1b;重则会直接触法律&#xff0c;受到主管部门的处罚和制…