iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >使用try-catch-finally常遇到的坑有哪些
  • 650
分享到

使用try-catch-finally常遇到的坑有哪些

2023-06-15 15:06:16 650人浏览 八月长安
摘要

本篇内容主要讲解“使用try-catch-finally常遇到的坑有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“使用try-catch-finally常遇到的坑有哪些”吧!坑1:final

本篇内容主要讲解“使用try-catch-finally常遇到的坑有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“使用try-catch-finally常遇到的坑有哪些”吧!

坑1:finally中使用return

若在 finally 中使用 return,那么即使 try-catch 中有 return 操作,也不会立马返回结果,而是再执行完 finally 中的语句再返回。

此时问题就产生了:如果 finally 中存在 return 语句,则会直接返回 finally 中的结果,从而无情的丢弃了 try 中的返回值。

① 反例代码

public static void main(String[] args) throws FileNotFoundException {      System.out.println("执行结果:" + test()); }  private static int test() {      int num = 0;      try {          // num=1,此处不返回          num++;          return num;      } catch (Exception e) {          // do something      } finally {          // num=2,返回此值          num++;          return num;      }  }

以上代码的执行结果如下:

使用try-catch-finally常遇到的坑有哪些

② 原因分析

如果在 finally 中存在 return 语句,那么 try-catch 中的 return 值都会被覆盖,如果程序员在写代码的时候没有发现这个问题,那么就会导致程序的执行结果出错。

③ 解决方案

如果 try-catch-finally 中存在 return 返回值的情况,一定要确保 return 语句只在方法的尾部出现一次。

④ 正例代码

public static void main(String[] args) throws FileNotFoundException {      System.out.println("执行结果:" + testAmend());  }  private static int testAmend() {      int num = 0;      try {          num = 1;      } catch (Exception e) {          // do something      } finally {          // do something      }      // 确保 return 语句只在此处出现一次      return num;  }

坑2:finally中的代码“不执行”

如果说上面的示例比较简单,那么下面这个示例会给你不同的感受,直接来看代码。

① 反例代码

public static void main(String[] args) throws FileNotFoundException {      System.out.println("执行结果:" + getValue());  }  private static int getValue() {      int num = 1;      try {         return num;      } finally {          num++;      }  }

以上代码的执行结果如下:

使用try-catch-finally常遇到的坑有哪些

② 原因分析

本以为执行的结果会是 2,但万万没想到竟然是 1,用马大师的话来讲:「我大意了啊,没有闪」。

有人可能会问:如果把代码换成 ++num,那么结果会不会是 2 呢?

很抱歉的告诉你,并不会,执行的结果依然是 1。那为什么会这样呢?想要真正的搞懂它,我们就得从这段代码的字节码说起了。

以上代码最终生成的字节码如下:

// class version 52.0 (52)  // access flags 0x21  public class com/example/basic/FinallyExample {    // compiled from: FinallyExample.java    // access flags 0x1    public <init>()V     L0      LINENUMBER 5 L0      ALOAD 0      INVOKESPECIAL java/lang/Object.<init> ()V      RETURN     L1      LOCALVARIABLE this Lcom/example/basic/FinallyExample; L0 L1 0      MAXSTACK = 1      MAXLOCALS = 1    // access flags 0x9    public static main([Ljava/lang/String;)V throws java/io/FileNotFoundException      L0      LINENUMBER 13 L0      GETSTATIC java/lang/System.out : Ljava/io/PrintStream;      NEW java/lang/StringBuilder      DUP      INVOKESPECIAL java/lang/StringBuilder.<init> ()V      LDC "\u6267\u884c\u7ed3\u679c:"      INVOKEVIRTUAL java/lang/StringBuilder.append (Ljava/lang/String;)Ljava/lang/StringBuilder;      INVOKESTATIC com/example/basic/FinallyExample.getValue ()I      INVOKEVIRTUAL java/lang/StringBuilder.append (I)Ljava/lang/StringBuilder;      INVOKEVIRTUAL java/lang/StringBuilder.toString ()Ljava/lang/String;      INVOKEVIRTUAL java/io/PrintStream.println (Ljava/lang/String;)V     L1      LINENUMBER 14 L1      RETURN     L2      LOCALVARIABLE args [Ljava/lang/String; L0 L2 0      MAXSTACK = 3      MAXLOCALS = 1    // access flags 0xA    private static getValue()I      TRYCATCHBLOCK L0 L1 L2 null     L3      LINENUMBER 18 L3      ICONST_1      ISTORE 0     L0      LINENUMBER 20 L0      ILOAD 0      ISTORE 1     L1      LINENUMBER 22 L1      IINC 0 1     L4      LINENUMBER 20 L4      ILOAD 1      IRETURN     L2      LINENUMBER 22 L2     FRAME FULL [I] [java/lang/Throwable]      ASTORE 2      IINC 0 1     L5      LINENUMBER 23 L5      ALOAD 2      ATHROW     L6      LOCALVARIABLE num I L0 L6 0      MAXSTACK = 1      MAXLOCALS = 3  }

这些字节码的简易版本如下图所示:

使用try-catch-finally常遇到的坑有哪些

想要读懂这些字节码,首先要搞懂这些字节码所代表的含义,这些内容可以从 oracle 的官网查询到(英文文档):https://docs.oracle.com/javase/specs/JVMs/se8/html/jvms-6.html

磊哥在这里对这些字节码做一个简单的翻译

iconst 是将 int 类型的值压入操作数栈。istore 是将 int 存储到局部变量。iload 从局部变量加载 int 值。iinc 通过下标递增局部变量。ireturn 从操作数堆栈中返回 int 类型的值。astore 将引用存储到局部变量中。

有了这些信息之后,我们来翻译一下上面的字节码内容:

0 iconst_1   在操作数栈中存储数值 1  1 istore_0   将操作数栈中的数据存储在局部变量的位置 0  2 iload_0    从局部变量读取值到操作数栈  3 istore_1   将操作数栈中存储 1 存储在局部变量的位置 1  4 iinc 0 by 1 把局部变量位置 0 的元素进行递增(+1)操作  7 iload_1 将局部位置 1 的值加载到操作数栈中  8 ireturn 返回操作数栈中的 int 值

通过以上信息也许你并不能直观的看出此方法的内部执行过程,没关系磊哥给你准备了方法执行流程图:

使用try-catch-finally常遇到的坑有哪些

使用try-catch-finally常遇到的坑有哪些

使用try-catch-finally常遇到的坑有哪些

通过以上图片我们可以看出:在 finally 语句(iinc 0, 1)执行之前,本地变量表中存储了两个信息,位置 0 和位置 1 都存储了一个值为 1 的 int 值。而在执行 finally(iinc 0, 1)之前只把位置 0 的值进行了累加,之后又将位置 1 的值(1)返回给了操作数栈,所以当执行返回操作(ireturn)时会从操作数栈中读到返回值为 1 的结果,因此最终的执行是 1 而不是 2。

③ 解决方案

关于 Java 虚拟机是如何编译 finally 语句块的问题,有兴趣的读者可以参考《The JavaTM Virtual Machine Specification, Second Edition》中 7.13 节 Compiling finally。那里详细介绍了 Java 虚拟机是如何编译 finally 语句块。

实际上,Java 虚拟机会把 finally 语句块作为 subroutine(对于这个 subroutine 不知该如何翻译为好,干脆就不翻译了,免得产生歧义和误解)直接插入到 try 语句块或者 catch 语句块的控制转移语句之前。但是,还有另外一个不可忽视的因素,那就是在执行 subroutine(也就是 finally 语句块)之前,try 或者 catch 语句块会保留其返回值到本地变量表(Local Variable Table)中,待 subroutine 执行完毕之后,再恢复保留的返回值到操作数栈中,然后通过 return 或者 throw 语句将其返回给该方法的调用者(invoker)。

因此如果在 try-catch-finally 中如果有 return 操作,**一定要确保 return 语句只在方法的尾部出现一次!**这样就能保证 try-catch-finally 中所有操作代码都会生效。

④ 正例代码

private static int getValueByAmend() {      int num = 1;      try {          // do something      } catch (Exception e) {          // do something      } finally {          num++;      }      return num;  }

坑3:finally中的代码“非最后”执行

① 反例代码

public static void main(String[] args) throws FileNotFoundException {      execErr();  }  private static void execErr() {      try {          throw new RuntimeException();     } catch (RuntimeException e) {          e.printStackTrace();      } finally {          System.out.println("执行 finally.");      }  }

以上代码的执行结果如下:

使用try-catch-finally常遇到的坑有哪些

从以上结果可以看出 finally 中的代码并不是最后执行的,而是在 catch 打印异常之前执行的,这是为什么呢?

② 原因分析

产生以上问题的真实原因其实并不是因为 try-catch-finally,当我们打开 e.printStackTrace 的源码就能看出一些端倪了,源码如下:

使用try-catch-finally常遇到的坑有哪些

从上图可以看出,当执行 e.printStackTrace()  和 finally 输出信息时,使用的并不是同一个对象。finally 使用的是标准输出流:System.out,而 e.printStackTrace()  使用的却是标准错误输出流:System.err.println,它们执行的效果等同于:

public static void main(String[] args) {      System.out.println("我是标准输出流");      System.err.println("我是标准错误输出流");  }

而以上代码执行结果的顺序也是随机的,而产生这一切的原因,我们或许可以通过标准错误输出流(System.err)的注释和说明文档中看出:

使用try-catch-finally常遇到的坑有哪些

使用try-catch-finally常遇到的坑有哪些

我们简单的对以上的注释做一个简单的翻译:

“标准”错误输出流。该流已经打开,并准备接受输出数据。通常,此流对应于主机环境或用户指定的显示输出或另一个输出目标。按照惯例,即使主要输出流(out 输出流)已重定向到文件或其他目标位置,该输出流(err 输出流)也能用于显示错误消息或其他信息,这些信息应引起用户的立即注意。

从源码的注释信息可以看出,标准错误输出流(System.err)和标准输出流(System.out)使用的是不同的流对象,即使标准输出流并定位到其他的文件,也不会影响到标准错误输出流。那么我们就可以大胆的猜测:二者是独立执行的,并且为了更高效的输出流信息,二者在执行时是并行执行的,因此我们看到的结果是打印顺序总是随机的。

为了验证此观点,我们将标准输出流重定向到某个文件,然后再来观察 System.err 能不能正常打印,实现代码如下:

public static void main(String[] args) throws FileNotFoundException {      // 将标准输出流的信息定位到 log.txt 中      System.setOut(new PrintStream(new FileOutputStream("log.txt")));      System.out.println("我是标准输出流");      System.err.println("我是标准错误输出流");  }

以上代码的执行结果如下:

使用try-catch-finally常遇到的坑有哪些

当程序执行完成之后,我们发现在项目的根目录出现了一个新的 log.txt 文件,打开此文件看到如下结果:

使用try-catch-finally常遇到的坑有哪些

从以上结果可以看出标准输出流和标准错误输出流是彼此独立执行的,且 JVM 为了高效的执行会让二者并行运行,所以最终我们看到的结果是 finally 在 catch 之前执行了。

③ 解决方案

知道了原因,那么问题就好处理,我们只需要将 try-catch-finally 中的输出对象,改为统一的输出流对象就可以解决此问题了。

④ 正例代码

private static void execErr() {      try {          throw new RuntimeException();      } catch (RuntimeException e) {          System.out.println(e);      } finally {          System.out.println("执行 finally.");      }  }

改成了统一的输出流对象之后,我手工执行了 n 次,并没有发现任何问题。

坑4:finally中的代码“不执行”

finally 中的代码一定会执行吗?如果是之前我会毫不犹豫的说“是的”,但在遭受了社会的毒打之后,我可能会这样回答:正常情况下 finally 中的代码一定会执行的,但如果遇到特殊情况 finally 中的代码就不一定会执行了,比如下面这些情况:

  •  在 try-catch 语句中执行了 System.exit;

  •  在 try-catch 语句中出现了死循环;

  •  在 finally 执行之前掉电或者 JVM 崩溃了。

如果发生了以上任意一种情况,finally 中的代码就不会执行了。虽然感觉这一条有点“抬杠”的嫌疑,但墨菲定律告诉我们,如果一件事有可能会发生,那么他就一定会发生。所以从严谨的角度来说,这个观点还是成立的,尤其是对于新手来说,神不知鬼不觉的写出一个自己发现不了的死循环是一件很容易的事,不是嘛?

① 反例代码

public static void main(String[] args) {      noFinally();  }  private static void noFinally() {      try {         System.out.println("我是 try~");          System.exit(0);      } catch (Exception e) {          // do something      } finally {          System.out.println("我是 fially~");     }  }

以上代码的执行结果如下:

使用try-catch-finally常遇到的坑有哪些

从以上结果可以看出 finally 中的代码并没有执行。

② 解决方案

排除掉代码中的 System.exit 代码,除非是业务需要,但也要注意如果在 try-cacth 中出现了 System.exit 的代码,那么 finally 中的代码将不会被执行。

到此,相信大家对“使用try-catch-finally常遇到的坑有哪些”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

--结束END--

本文标题: 使用try-catch-finally常遇到的坑有哪些

本文链接: https://www.lsjlt.com/news/280742.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 使用try-catch-finally常遇到的坑有哪些
    本篇内容主要讲解“使用try-catch-finally常遇到的坑有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“使用try-catch-finally常遇到的坑有哪些”吧!坑1:final...
    99+
    2023-06-15
  • Try-Catch-Finally中的坑有哪些
    这篇文章主要讲解了“Try-Catch-Finally中的坑有哪些”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Try-Catch-Finally中的坑有哪...
    99+
    2024-04-02
  • PHP异常处理:如何使用try-catch-finally语句
    在php中,try-catch-finally语句用于异常处理,通过保护代码块并提供异常处理和清理机制来增强应用程序的健壮性。 PHP异常处理:使用try-catch-finally语...
    99+
    2024-05-14
    php 异常处理 mysql
  • 为什么不使用try-catch-finally处理Java异常
    本篇内容主要讲解“为什么不使用try-catch-finally处理Java异常”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“为什么不使用try-catch-finally处理Java异常”吧!...
    99+
    2023-06-15
  • C#中怎么使用try-catch-finally结构处理异常
    在C#中,可以使用try-catch-finally结构来处理异常。try块中包含可能会引发异常的代码,catch块用于捕获并处理异...
    99+
    2024-03-06
    C#
  • BigDecimal遇到的坑有哪些
    这篇文章主要介绍“BigDecimal遇到的坑有哪些”,在日常操作中,相信很多人在BigDecimal遇到的坑有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”BigDec...
    99+
    2024-04-02
  • 使用recvfrom函数遇到的坑有哪些
    使用recvfrom函数时可能会遇到以下几个常见的问题:1. 阻塞问题:recvfrom函数默认是阻塞的,即当没有收到数据时,程序会...
    99+
    2023-08-24
    recvfrom
  • JavaScript try...catch语句的常见错误:千万别踩这些坑
    1. 不捕获错误 最常见的错误之一就是根本不捕获错误。例如,以下代码没有使用try...catch语句来捕获错误: function divide(a, b) { return a / b; } divide(10, 0); // ...
    99+
    2024-02-23
    JavaScript try...catch 语句 错误处理 常见错误
  • 使用RxJava中遇到的一些”坑“
    前言大家越用RxJava,越觉得它好用,所以不知不觉地发现代码里到处都是RxJava的身影。然而,RxJava也不是银弹,其中仍然有很多问题需要解决。这里,我简单地总结一下自己遇到的一些“坑”,内容上可能会比较松散。一、考虑主线程的切换Rx...
    99+
    2023-05-31
    rxjava 使用 ava
  • Android Studio升级到3.0后遇到的坑有哪些
    这篇文章主要介绍Android Studio升级到3.0后遇到的坑有哪些,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!第一坑、必须升级gradle到4.0以上相信这个大坑,一般使用as的人都会解决了,所以就不多说第二...
    99+
    2023-05-30
    android studio
  • Java异常处理try catch的基本使用
    目录1. 异常1.1 try…catch异常处理1.2 多catch并行处理1.3 throw和throws 关键字的使用1.4 finally代码块1.5 Runti...
    99+
    2024-04-02
  • JavaScript中不要在循环内部使用try-catch-finally的示例分析
    这篇文章主要为大家展示了“JavaScript中不要在循环内部使用try-catch-finally的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Ja...
    99+
    2024-04-02
  • vue resource post请求时遇到的坑有哪些
    这篇文章将为大家详细讲解有关vue resource post请求时遇到的坑有哪些,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。使用 post 请求// glo...
    99+
    2024-04-02
  • js中的异常处理try...catch使用介绍
    在JavaScript可以使用try...catch来进行异常处理。例如: 复制代码 代码如下: try { foo.bar();} catch (e) { alert(e.name...
    99+
    2022-11-15
    异常处理 try catch
  • Spring中@RequestParam使用及遇到的一些坑
    目录加与不加的区别使用RequestParam遇到的一些坑(总结)总结加与不加的区别 @RequestMapping("/list1") public String test1(in...
    99+
    2024-04-02
  • Java深入讲解异常处理try catch的使用
    目录1.try-catch异常处理说明2.try-catch异常处理细节示例01测试结果01示例02测试结果021.try-catch异常处理说明 Java提供try和catch块来...
    99+
    2024-04-02
  • mysql中timestamp比较查询遇到的坑有哪些
    这篇文章主要介绍mysql中timestamp比较查询遇到的坑有哪些,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!timestamp比较查询遇到的坑要求mysql建表的时候update_time 为timestamp...
    99+
    2023-06-21
  • 如何使用JavaScript中的try catch throw处理异常
    这篇文章主要为大家展示了“如何使用JavaScript中的try catch throw处理异常”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“如何使用JavaS...
    99+
    2024-04-02
  • Linux实践中使用重定向和管道符遇到的坑有哪些
    这篇文章给大家介绍Linux实践中使用重定向和管道符遇到的坑有哪些,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。我很喜欢 Linux 系统,尤其是 Linux  的一些设计很漂亮,比如可以将一些复杂的问题分解...
    99+
    2023-06-15
  • 微信JS-SDK选择图片遇到的坑有哪些
    这篇文章主要介绍了微信JS-SDK选择图片遇到的坑有哪些,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。有个需求要在微信企业号里面做开发,有个...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作