关于Gen生成try-catch-finally
例1:
class TestExc extends Exception{}
void tryItOut () throws TestExc{}
void handleExc(Object o){}
void catchOne() {
try {
tryItOut();
} catch (TestExc e) {
handleExc(e);
}
}
生成的class文件的格式如下:
void catchOne();
flags:
Code:
stack=2, locals=2, args_size=1
0: aload_0
1: invokevirtual #2 // Method tryItOut:()V
4: goto 13
7: astore_1
8: aload_0
9: aload_1
10: invokevirtual #4 // Method handleExc:(Ljava/lang/Object;)V
13: return
Exception table:
from to target type
0 4 7 Class com/test19/Test04$TestExc
其中的catchOne()方法的code码如下:
2A B6 00 02 A7 00 09 4C 2A 2B B6 00 04 B1 opcode [2A] - 0000: aload_0 opcode [B6] - 0001: invokevirtual 2 [Methodref: com.test19.Test04.tryItOut, parameter = (), returns = void] opcode [A7] - 0004: goto 9 opcode [4C] - 0007: astore_1 opcode [2A] - 0008: aload_0 opcode [2B] - 0009: aload_1 opcode [B6] - 0010: invokevirtual 4 [Methodref: com.test19.Test04.handleExc, parameter = (java.lang.Object), returns = void] opcode [B1] - 0013: return
例2:
如上的例子与如下的例子生成的class文件一样,但是Exception Table不一样:
class TestExc1 extends Exception{}
class TestExc2 extends Exception{}
void tryItOut () throws TestExc1,TestExc2{}
void handleExc1(Object o){}
void handleExc2(Object o){}
void nestedCatch() {
try {
tryItOut();
} catch (TestExc1 e) {
handleExc1(e);
} catch (TestExc2 e) {
handleExc2(e);
}
}
生成的class文件的格式如下:
void nestedCatch();
flags:
Code:
stack=2, locals=2, args_size=1
0: aload_0
1: invokevirtual #2 // Method tryItOut:()V
4: goto 22
7: astore_1
8: aload_0
9: aload_1
10: invokevirtual #4 // Method handleExc1:(Ljava/lang/Object;)V
13: goto 22
16: astore_1
17: aload_0
18: aload_1
19: invokevirtual #6 // Method handleExc2:(Ljava/lang/Object;)V
22: return
Exception table:
from to target type
0 4 7 Class com/test19/Test07$TestExc1
0 4 16 Class com/test19/Test07$TestExc2
再举个例子,如下:
void tryItOut () throws TestExc1,TestExc2{}
class TestExc1 extends Exception{}
class TestExc2 extends Exception{}
void handleExc1(Object o){}
void handleExc2(Object o){}
void nestedCatch() {
try {
try {
tryItOut();
} catch (TestExc1 e) {
handleExc1(e);
}
} catch (TestExc2 e) {
handleExc2(e);
}
}
生成的class文件内容如下:
void nestedCatch();
flags:
Code:
stack=2, locals=2, args_size=1
0: aload_0
1: invokevirtual #2 // Method tryItOut:()V
4: goto 13
7: astore_1
8: aload_0
9: aload_1
10: invokevirtual #4 // Method handleExc1:(Ljava/lang/Object;)V
13: goto 22
16: astore_1
17: aload_0
18: aload_1
19: invokevirtual #6 // Method handleExc2:(Ljava/lang/Object;)V
22: return
Exception table:
from to target type
0 4 7 Class com/test19/Test04$TestExc1
0 13 16 Class com/test19/Test04$TestExc2
例3:
void tryItOut () {}
void wrapItUp () {}
void tryFinally() {
try {
tryItOut();
} finally {
wrapItUp();
}
}
生成的class文件格式如下:
void tryFinally();
flags:
Code:
stack=1, locals=2, args_size=1
0: aload_0
1: invokevirtual #2 // Method tryItOut:()V
4: aload_0
5: invokevirtual #3 // Method wrapItUp:()V
8: goto 18
11: astore_1
12: aload_0
13: invokevirtual #3 // Method wrapItUp:()V
16: aload_1
17: athrow
18: return
Exception table:
from to target type
0 4 11 any
11 12 11 any
例4:
int tryFinally() {
try {
int i = 1;
return i;
} finally {
int j = 2;
return j;
}
}
生成的class文件内容如下:
int tryFinally();
flags:
Code:
stack=1, locals=6, args_size=1
0: iconst_1
1: istore_1
2: iload_1
3: istore_2 // 由于要运行finally,所以需要将这个return的值暂时存储到本地变量表中
4: iconst_2
5: istore_3
6: iload_3
7: ireturn
8: astore 4 // 将异常引用类型存储到本地变量表4的位置
10: iconst_2
11: istore 5
13: iload 5 // 将本地变量表5的位置加载到栈中,因为finally中的return值要做为最终值返回
15: ireturn
Exception table:
from to target type
0 4 8 any
8 10 8 any
例5:
void tryItOut () {}
void wrapItUp1 () {}
void wrapItUp2 () {}
int tryFinally() {
try {
try {
int i = 1;
return i;
} finally {
wrapItUp1();
}
} finally {
wrapItUp2();
}
}
则生成的代码如下:
stack=1, locals=5, args_size=1
0: iconst_1
1: istore_1
2: iload_1
3: istore_2 // 将变量i的值存储到本地变量表2的位置,以便返回
4: aload_0
5: invokevirtual #2 // Method wrapItUp1:()V
8: aload_0
9: invokevirtual #3 // Method wrapItUp2:()V
12: iload_2 // 从本地变量表2的位置取出返回数并返回
13: ireturn
14: astore_3 // 将异常存储到本地变量表3的位置
15: aload_0
16: invokevirtual #2 // Method wrapItUp1:()V
19: aload_3
// 在这里抛出异常后,在异常表中查看跳转到target,
// 则清空当前操作数栈,异常重新入栈,程序继续执行
20: athrow
21: astore 4 // 将异常存储到本地变量表4的位置
23: aload_0
24: invokevirtual #3 // Method wrapItUp2:()V
27: aload 4
29: athrow
Exception table:
from to target type
0 4 14 any
14 15 14 any
0 8 21 any
14 23 21 any
例6:
class TestExc1 extends Exception{}
class TestExc2 extends Exception{}
void tryItOut () throws TestExc1,TestExc2{}
void handleExc1(Object o){}
void handleExc2(Object o){}
void wrapItUp () {}
void nestedCatch() {
try {
tryItOut();
} catch (TestExc1 e) {
handleExc1(e);
} catch (TestExc2 e) {
handleExc2(e);
}finally {
wrapItUp();
}
}



关于Gen生成try-catch-finally的更多相关文章
- idea 自动生成try/catch代码块的快捷键
好像每个人的快捷键可能不同:我的是 Alt+Shift+Z 网上查的是 Ctrl+Alt+T 如果都不是可以点选工具栏生成try/catch(并可查看到自己的快捷键是什么):Code->Su ...
- C#生成带项目编号的Word段落
using System; using Microsoft.Office.Interop.Word; using Word = Microsoft.Office.Interop.Word; names ...
- Eclipse用法和技巧八:自动添加try/catch块1
站在编译器的角度来看,java中的异常可以分为两种,已检查异常和未检查异常.对于已检查异常比如IO操作,编译器会要求设置try/catch语句块,在eclipse中也只要使用帮助快捷键ctrl+1,就 ...
- java生成二维码(最初版)
研究了2个小时,发现自己竟然智障,用原先的图片覆盖另一个图片 package com.tz.util; import java.awt.Color;import java.awt.Graphics2D ...
- 用SAX和PULL进行XML文件的解析与生成
XML解析有传统的dom方法还有Jsoup,SAX,PULL等,这里讲的是比较省内存的SAX和PULL方法.Android中极力推荐用PULL的方式来解析,我个人觉得pull确实比较简单,但其内部的逻 ...
- java图片裁剪和java生成缩略图
一.缩略图 在浏览相冊的时候.可能须要生成相应的缩略图. 直接上代码: public class ImageUtil { private Logger log = LoggerFactory.getL ...
- java使用freemarker生成静态html页面
1. 模板文件static.html <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" " ...
- Android 如何修改gen下包的名字
前言 当将项目中包进行重命名后,所有包名字修改了,但是在gen目录下android sdk 自动生成的包名并没有修改,如果要引用R.java 中的字段, 又得import以前的包名字. 原因 出现 ...
- JAVA使用qrcode生成二维码(带logo/不带logo)
/** * */ package qrcode; import java.awt.Color; import java.awt.Graphics2D; import java.awt.Image; i ...
随机推荐
- struts2 一些注解
实现的JSP页面位置 web-root/jsp/user/add.jsp /update.jsp // /* @Namespace("/t") @AllowedMethods(va ...
- (匹配)Dolls --HDU --4160
链接: http://acm.hdu.edu.cn/showproblem.php?pid=4160 代码: #include<stdio.h> #include<string.h& ...
- arduino远程刷新(烧录)固件
在车间部署了十几个网络版的温湿度采集器(基于arduino的),这些采集器分布在不同的地方,现在要更新一下上面的固件.最笨的方法是一个一个地取下来,插到电脑的USB接口上进行固件更新,这样做显然很麻烦 ...
- 使用MEF与Castle实现AOP
MEF是微软的一个ioc框架,使用非常方便,我们只需要在需要导出的类上标记[Export],在需要使用的地方[import]就可以使用了.现在我们扩展MEF,在其装配生成实例时,使用Castle Dy ...
- javascript js 完美解决 click 与 dblclick 冲突,并且不会导致click延迟
示例代码: marker.addEventListener("click", function(){ if (!window.markerClicked) { window.mar ...
- Chrome浏览器插件开发-淘宝自动登录
浏览器插件的介绍 Chrome浏览器插件开发的准备工作 manifest.json配置介绍 页面如何注入scripts文件 一. 浏览器插件的介绍 浏览器插件是一种遵循一定规范的应用程序接口编写出来的 ...
- 基于Spring Boot的Logback日志轮转配置
在生产环境下,日志是最好的问题调试和跟踪方法,因此日志的地位是十分重要的.我们平时经常使用的log4j,slf4j,logback等等,他们的配置上大同小异.这里就结合Spring Boot配置一下L ...
- 微信小程序与vueJs的异同
简而言之,所有的框架都是建立在原生javascript基础之上的,所以对于有一定js基础的同学来说,各种框架都是比较容易入手的,但不同的框架之间又有一定的差别,有时候切换使用时就会掉入坑了. 一.微信 ...
- webpack快速入门——Json配置文件使用
在实际工作中,我们的项目都会配置一个Json的文件或者说API文件,作为项目的配置文件. 有时候你也会从后台读取到一个json的文件,这节课就学习如何在webpack环境中使用Json. 如果你会we ...
- Android多媒体整体架构图
Android多媒体整体架构图 MediaPlayer框架图 Camera框架图 SoundRecorder框架图 VideoCamera框架图 OpenCore与Skia ALSA Audio框架图 ...