Java Magic. Part 3: Finally
Java Magic. Part 3: Finally
@(Base)[JDK, magic, 黑魔法]
转载请写明:原文地址
英文原文
系列文章:
-Java Magic. Part 1: java.net.URL
-Java Magic. Part 2: 0xCAFEBABE
-Java Magic. Part 3: Finally
-Java Magic. Part 4: sun.misc.Unsafe
所有JAVA程序员都应该知道一个基本概念-finally一定会被执行,但是真的是这样么?
这取决于“执行”是什么意思。但总的来说答案都是,YES。
Normal program execution
有人可能拿如下的例子来反驳:
try {
System.exit(1);
} finally {
System.out.println("I'm here, man");
}
刚才是不是你说的finlly块一定会被执行。
显然在上面这个例子中,Sout
语句不会被执行。我们刚才也说了,是正常的程序六层中答案是肯定的。
下面这句话来自官方文档
Note: If the JVM exits while the try or catch code is being executed, then the finally block may not execute.
我们再看下面这段代码,第二行是不是一定会执行呢?
Note: If the JVM exits while the try or catch code is being executed, then the finally block may not execute.
答案仍然是肯定的。除非...BANG...断电了,程序终止。
这是神马意思呢?意思就是程序非正常的运行。我们没办法保证所有事情。显然,这和System.exit(1)
是一个意思,也和你电脑上的重启键是一个意思。
所以我们不讨论这种情况啦。没劲。
Perpetuum Mobile
永动机 我们再来看如下代码:
try {
while (true) {
System.out.println("I print here some useful information");
}
} finally {
System.out.println("Let me run");
}
这里的finlly块会执行吗?当然可能,就是在标准输出异常的时候。但是绝大部分情况答案仍然是不会。
Threads
什么是线程?我们都知道线程的执行流程可以被interrupted。
假设我们有一个线程正在执行一些东西,另外一个线程kill掉当前线程(此时他正要执行finally)。
还有一个场景是,如果两个线程之间有死锁,那么还是不会执行finally块的。
下面这段话仍然来自官方文档
...if the thread executing the try or catch code is interrupted or killed, the finally block may not execute even though the application as a whole continues.
所以这种情况下,我们可以把线程当做一个独立的程序来看:
Finally块肯定是会执行的。除了程序或者线程异常的终止了。
Finally we return
好了,根据上述的文字,我们了解了finally一定会被执行。但是你知道finally什么时候会被执行吗?
考虑如下代码:
int someFunc() {
try {
return 0;
} finally {
return 1;
}
}
这个返回值是多少呢? 返回值是1
。因为finally
一定会被执行。
考虑下面这段呢?
int someFunc() {
try {
throw new RuntimeException();
} finally {
return 1;
}
}
答案仍然是1。但是有个问题是,异常就被吞掉了。这个场景是一个非常有名的场景,叫做exception swallowing
。这是一个非常危险示例,因为client的代码以为你会返回一个值或者抛出一个异常,但是你永远都只是返回一个值。
下面我们最后看一个类似的例子:
String deposit(int amount) throws DAOException {
try {
return dao.deposit(amount);
} finally {
return "OK";
}
}
dao.deposit
会抛出一个受检的异常,导致客户端必须处理这个异常,但是呢,由于上面我们已经提到的原因,这个函数又永远返回OK
,是不是有点淡淡的忧伤。
所以我们得到另外一个结论:
永远不要在finally
里面使用return
语句。
Instead of conclusion
很多程序员已经意识到finally的一些问题。但根据我们上面说的,只需要注意两点就可以避免犯错~
- Rule 1 Finally executes always, except the case where controlling program or thread was aborted.
- Rule 2 Never use return from finally block.
Java Magic. Part 3: Finally的更多相关文章
- Java Magic. Part 4: sun.misc.Unsafe
Java Magic. Part 4: sun.misc.Unsafe @(Base)[JDK, Unsafe, magic, 黑魔法] 转载请写明:原文地址 系列文章: -Java Magic. P ...
- Java Magic. Part 2: 0xCAFEBABE
Java Magic. Part 2: 0xCAFEBABE @(Base)[JDK, magic, 黑魔法] 转载请写明:原文地址 英文原文 系列文章: -Java Magic. Part 1: j ...
- Java Magic. Part 1: java.net.URL
Java Magic. Part 1: java.net.URL @(Base)[JDK, url, magic, 黑魔法] 英文原文 转载请写明:原文地址 系列文章: -Java Magic. Pa ...
- Magic Cast Method in Java Magic Trick In Java
https://www.atlassian.com/blog/archives/magic_trick_in_java https://www.gamlor.info/wordpress/2010/1 ...
- 【Java并发】详解 AbstractQueuedSynchronizer
前言 队列同步器 AbstractQueuedSynchronizer(以下简称 AQS),是用来构建锁或者其他同步组件的基础框架.它使用一个 int 成员变量来表示同步状态,通过 CAS 操作对同步 ...
- 【java编程】Java魔法类:Unsafe应用解析
转载来源:https://tech.meituan.com/2019/02/14/talk-about-java-magic-class-unsafe.html 前言 Unsafe是位于sun.mis ...
- Java sun.misc.unsafe类
Java是一个安全的开发工具,它阻止开发人员犯很多低级的错误,而大部份的错误都是基于内存管理方面的.如果你想搞破坏,可以使用Unsafe这个类.这个类是属于sun.*API中的类,并且它不是J2SE中 ...
- Java并发编程-Unsafe实现原理与Unsafe应用解析
前言 Unsafe是位于sun.misc包下的一个类,主要提供一些用于执行低级别.不安全操作的方法,如直接访问系统内存资源.自主管理内存资源等,这些方法在提升Java运行效率.增强Java语言底层资源 ...
- sun.misc.unsafe
Java中大部分错误都是基于内存管理方面的.如果想破坏,可以使用Unsafe这个类. 实例化Unsafe: 下面两种方式是不行的 private Unsafe() {} //私有构造方法 @Calle ...
随机推荐
- dubbo实战
http://blog.csdn.net/hejingyuan6/article/details/47403299
- python初始环境安装
Python下载地址 Python官网:https://www.python.org/ 在该网可以下载Python最新及历史版本.可以下载基于Windows或其它操作系统的版本. Python安装 本 ...
- Druid 连接池 JDBCUtils 工具类的使用
Druid工具介绍 它不仅仅是一个数据库连接池,它还包含一个ProxyDriver,一系列内置的JDBC组件库,一个SQL Parser. 支持所有JDBC兼容的数据库,包括Oracle.MySQL. ...
- 学习笔记之IKM C++ 11
https://github.com/haotang923/interview/tree/master/IKM Q1. If most of the calls to function foo() b ...
- MySQL excel导入
说明: 1 因在测试发现如果用SQLyog导入数据需要下载excel驱动,因而选择Navicat 2 之前选择excel文件为xlsx 发现Navicat识别不了,因而转存为xls文件,测试OK 1 ...
- windows,linux下SVN实现自动更新WEB目录
通过SVN进行版本库管理,每次提交后,都要在SVN服务器更新最新上传的版本到WEB目录进行同步.操作比较烦琐,而且效率也低.使用SVN钩子脚本进行WEB目录同步,可很好的解决这方面的问题.由于测试机器 ...
- Unreal Engine 4 性能优化工具(Profiler Tool)
转自:http://aigo.iteye.com/blog/2296548 Profiler Tool Reference https://docs.unrealengine.com/latest/I ...
- Jade模板引擎使用详解
在 Express 中调用 jade 模板引擎 jade 变量调用 if 判断 循环 Case 选择 在模板中调用其他语言 可重用的 jade 块 (Mixins) 模板包含 (Includes) 模 ...
- Linux系统下MongoDB的简单安装与基本操作
这篇文章主要介绍了Linux系统下MongoDB的简单安装与基本操作,需要的朋友可以参考下 Mongo DB ,是目前在IT行业非常流行的一种非关系型数据库(NoSql),其灵活的数据存储方式,备 ...
- 代码生成器 CodeSmith 的使用(一)
由于在项目中经常要会用到数据库的 CRUD 操作(增.删.改.查),而且还使用的是orm 框架将数据库表名和表中的的字段映射成相应的类属性.如果把大量的时间用到手工输入数据库表中的字段,除了能练习打字 ...