本文获得 baeldung team 的翻译许可java
英文原文:www.baeldung.com/java-finall…git
在本教程中,咱们将研究 Java 中的 finally 关键字的用法。 咱们将看到如何在错误处理中与 try / catch 块一块儿使用它。 尽管 finally 的目的是保证代码被执行,可是咱们还将讨论 JVM 不执行 finally 代码的特殊状况。github
咱们还将讨论一些常见的陷阱,在这些陷阱中,finally 块可能会产生意外的结果。ide
try 关键字最后能够定义 finally 代码块。 finally 块中定义的代码,老是在 try 和任何 catch 块以后、方法完成以前运行。函数
正常状况下,无论是否抛出或捕获异常 finally 块都会执行。编码
try {
System.out.println("The count is " + Integer.parseInt(count));
} catch (NumberFormatException e) {
System.out.println("No count");
} finally {
System.out.println("In finally");
}
复制代码
在这个示例中,无论参数的值是多少,JVM 都执行 finally 块并输出“ In finally”。spa
try {
System.out.println("Inside try");
} finally {
System.out.println("Inside finally");
}
复制代码
结果操作系统
Inside
try Inside
finally线程
由于无论是否发生异常 finally 都会执行,所以咱们能够在 finally 代码块中执行关闭链接、关闭文件和释放线程的的操做。翻译
当 try 代码块执行完成, finally 代码块就能够执行,哪怕没有发生异常。
try {
System.out.println("Inside try");
} finally {
System.out.println("Inside finally");
}
复制代码
Inside try Inside finally
哪怕 异常没有被 catch , finally 代码块依然会执行。
try {
System.out.println("Inside try");
throw new Exception();
} finally {
System.out.println("Inside finally");
}
复制代码
即便出现未被处理的异常,JVM 依然会执行 finally 代码块的代码。
Inside try Inside finally Exception in thread "main" java.lang.Exception
try 代码块发生异常, 被 catch 捕捉, finally 依然会执行。
try {
System.out.println("Inside try");
throw new Exception();
} catch (Exception e) {
System.out.println("Inside catch");
} finally {
System.out.println("Inside finally");
}
复制代码
Inside try Inside catch Inside finally
即便 try 代码块中返回,也不能阻止 finally 代码块的执行。
try {
System.out.println("Inside try");
return "from try";
} finally {
System.out.println("Inside finally");
}
复制代码
JVM 会在返回到调用函数前执行 finally 代码块。
Inside try Inside finally
在 catch 代码块中添加返回语句,finally 代码依然会执行。
try {
System.out.println("Inside try");
throw new Exception();
} catch (Exception e) {
System.out.println("Inside catch");
return "from catch";
} finally {
System.out.println("Inside finally");
}
复制代码
结果
Inside try Inside catch Inside finally
尽管一般编写 finally 代码块是为了这段代码必定被执行到,可是也有一些特殊状况会致使 JVM 不会执行 finally 代码块。
若是操做系统中断了咱们的程序,那么finally 代码块可能就不能被执行。也有不少其余相似的行为致使 finally代码块不被执行。
try {
System.out.println("Inside try");
System.exit(1);
} finally {
System.out.println("Inside finally");
}
复制代码
结果
Inside try
try {
System.out.println("Inside try");
Runtime.getRuntime().halt(1);
} finally {
System.out.println("Inside finally");
}
复制代码
Inside try
若是守护线程刚开始执行到 finally 代码块,此时没有任何其余非守护线程,那么虚拟机将退出,此时 JVM 不会等待守护线程的 finally 代码块执行完成。
Runnable runnable = () -> {
try {
System.out.println("Inside try");
} finally {
try {
Thread.sleep(1000);
System.out.println("Inside finally");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
Thread regular = new Thread(runnable);
Thread daemon = new Thread(runnable);
daemon.setDaemon(true);
regular.start();
Thread.sleep(300);
daemon.start();
复制代码
输出
Inside try Inside try Inside finally
try {
System.out.println("Inside try");
while (true) {
}
} finally {
System.out.println("Inside finally");
}
复制代码
Try 代码块出现无限循环,且不出现异常,finally 也将永远得不到执行。
咱们在使用 finally 关键字时会遇到不少陷阱。
有一些很差的编码方式,如在 finally 代码块中存在返回值或者扔出异常。
finally 代码块包含返回语句,没有处理未捕获的异常。
try {
System.out.println("Inside try");
throw new RuntimeException();
} finally {
System.out.println("Inside finally");
return "from finally";
}
复制代码
此时,try 代码块中的 RuntimeException 会被忽略,函数返回 "from finally"字符串。
若是 finally 代码块中存在返回语句,则 try 和 catch 代码块若是存在返回语句就会被忽略。
try {
System.out.println("Inside try");
return "from try";
} finally {
System.out.println("Inside finally");
return "from finally";
}
复制代码
此段代码老是返回 “from finally” 。
若是再 finally 代码块中扔出异常,则 try 和 catch 中的异常扔出或者返回语句都将被忽略。
try {
System.out.println("Inside try");
return "from try";
} finally {
throw new RuntimeException();
}
复制代码
这段代码永远都不会有返回值,老是会抛出 RuntimeException。
本文咱们讨论了 Java 的 finally 关键字的用法。而后讨论了 finally 执行和不执行 finally 代码块的状况。
最后给出了开发中关于 finally 常见的使用的陷阱。
须要本文代码,能够去GitHub 配套 项目中下载。
译者补充: 结合第 4 部分的示例,你们能够思考一下,其余方式可让 finally 得不到执行吗? 欢迎你们在下方评论探讨。
若是你以为本文对你有帮助,欢迎点赞、转发、评论,你的支持是我创做的最大动力。