手记

(一)异常处理机制详解

# 前言

  本文主要是对Java异常处理机制的阐述,了解Java的异常机制的设计和分类,及Java异常有哪些坑,如何在自定义异常类时避免采坑。

# 异常机制分类

 

  异常情况是指阻止当前方法或作用域继续继续执行的情况。在Java中异常也是对象,我们可以像创建其他对象一样,用new在堆上创建异常对象。
从上图可以看到Throwable是所有异常类型的根类,它有两个重要的子类:Exception和Error。

  •  Error(错误)

  Error表示编译时和系统错误(除特殊情况外我们无需关注),比如代码允许是JVM运行错误,或内存不足时OutOfMemoryError。

  •  Exception(异常)

  Exception是可以抛出/处理的异常。在Java类库、用户方法及运行时故障都可能抛出Exception类型异常,我们程序员需要关注的主要是Exception。它又分为运行时异常和非运行时异常。

  运行时异常:由RuntimeException和其子类异常组成。比如NullPointerException(空指针异常)、IndexOutOfBoundsException(下标越界异常)。这些异常通常是非受检异常,可以捕获处理或者不处理。一般有程序逻辑引起的。运行时异常的特点是Java编译器编译时不会检查它,就算有这种异常编译也能通过,究其原因,RuntimeException代表的是编程错误。

  非运行时异常:包括RuntimeException以外的异常,类型上都属于Exception类及其子类。从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如IOException、SQLException等以及用户自定义的Exception异常。

# try-catch-finally捕获异常

  在Java中使用try-catch或者try-catch-finally捕获异常。

## try块

  对于有可能出现异常情况的代码块Code,我们可以把它放在try块里。

?

123try {//可能发生异常的代码块}

## 异常处理程序

  异常处理程序必须紧跟在try块后,以关键字catch表示。当异常被抛出时,异常处理机制将负责搜寻参数与异常类型相匹配的第一个程序,然后进入catch子程序执行。

?

 try {//可能发生异常的代码块} catch (Type1 id1) {//捕获并处理异常类型为Type1的异常} catch (Type2 id2) {//捕获并处理异常类型为Type1的异常} finally {//无论如何都会走到的代码//有如下极端情况不会走到finally代码块,但一般不考虑//比如CPU掉电、线程异常终止等} // etc...

  有时也可以采用maltiple catch。

## throw和throws

  我们在编程时,需要针对某种异常情况抛出异常给客户端,代码如下

?

 if (s == null) {throw new NullPointerException();}

  throws是一种“异常说明”方式,它属于方法声明的一部分,跟在形式参数列表之后。

?

1void func() throws Exception1, Exception2 {}

  这种异常说明的方式,可以强制函数使用者强制处理该异常情况。在定义抽象基类和接口时这种能力很重要,这样派生类就可以处理这些预先声明的异常。

  从上面可以看出throw主要是用来中断程序执行并移交异常对象到运行时处理。throws用于声明方法可抛出的异常,是异常说明的一种机制。

## 使用finally做清理工作

  对于一些代码,希望无论try块是否有异常抛出,都能得到执行,比如打开的文件句柄或者网络连接,可以使用try-catch-finally,代码如下所示。

public class FinallyWorks {    static int count = 0;    public static void main(String[] args) {        while (true) {            try {                // count为0时抛异常
                if (count ++ == 0) {                    throw new IOException();
                }
                System.out.println("No exception");
            } catch (Exception e) {//该句可以捕获所有异常
                System.out.println("IOException");
            } finally {
                System.out.println("In finally clause");                if (count == 2) break;
            }
        }
    }
}

/*** Output
**/IOException
In finally clause
No exception
In finally clause

## 新特性

### multiple exception

  如果一个try块中有多个异常要被捕获,catch块中的代码会变丑陋的同时还要用多余的代码来记录异常。有鉴于此,Java 7的一个新特征是:一个catch子句中可以捕获多个异常。示例代码如下:

catch(IOException | SQLException | Exception ex){
  log.warn(ex);
  throw new MyException(ex.getMessage());
}

### try-with-resources

  try-with-resources[1][2] 语句会确保在try语句结束时关闭所有资源。实现了java.lang.AutoCloseable或java.io.Closeable的对象都可以做为资源。使用try-with-resources进行资源的自动关闭,在try子句中能创建一个资源对象,当程序的执行完try-catch之后,运行环境自动关闭资源。示例代码如下:

/*** code 1
**/try (FileInputStream fis = new FileInputStream("example.java")) { // line 1
     System.out.println("fis created in try-with-resources");
     doSomething(); // line2} catch (Exception e) {
     e.printStackTrace();
}

  在Java7之前我们使用finally进行资源的关闭,如下所示

/*** code 2
**/FileInputStream fis = new FileInputStream("example.java");try {
     System.out.println("fis created in try-with-resources");
     doSomething(); // line 3} catch (Exception e) {
     e.printStackTrace();
} finally {    if (fis != null) {
        fis.close();// line 4    }
}

异常屏蔽
请参见参考文献[1][2]

# 正确的使用异常

## 不要在finally中使用return关键字。

  finally块中return返回后方法结束执行,会覆盖try块中的return语句,换句话说就是屏蔽了try块中的return语句。

/**
 * @author liangk
 * @date 18/09/2018 */public class FinallyReturn {    public static void main(String[] args) {
        String result = finallyReturnTest();
        System.out.println(result);

    } 

    public static String finallyReturnTest() {        try {
            System.out.println("finallyReturnTest start");
            String result = "Hello EveryBody!";            return result;
        } finally {            return "The finally block will be printed in the end";
        }
    }
}

/**
 * Output */finallyReturnTest start
The finally block will be printed in the end

## finally 块必须对资源对象、流对象进行关闭。

  如果JKD7及以上版本,可以使用上文介绍的try-with-resources方式

 

## 避免直接抛出RuntimeException及其子类。

  更不允许抛出Exception或Throwable(建议抛出具体的异常对象)

## 建议采用预检查方式规避RuntimeException异常,而不应该catch的方式处理。

public void readPreferences(String fileName) {
    InputStream in = new FileInputStream(fileName);
}

  上面的程序如果fileName是null,就会抛出NullPointerException,由于没有第一时间暴露问题,堆栈信息费解,需要相对复杂的定位。如果我们采取下面的方式,就很容易解决问题

public void readPreferences(String fileName) {
    Objects.requireNonNull(fileName);
    InputStream in = new FileInputStream(fileName);
}

## 不允许直接吞没异常。

  直接吞没异常,既不处理也不抛出,可能会导致难以诊断的异常情况,无法判断异常从哪里结束,什么原因产生的异常情况。

## try块只包含可能会出现异常的必要代码段。

  • try-catch 代码段会产生额外的性能开销,或者换个角度说,它往往会影响JVM对代码进行优化,所以建议仅捕获必要的代码段,不能包住整段代码。

  • - Java每实例化一个Exception,都会对当时的栈进行快照,这是一个相对较重的操作。如果异常频繁发生,开销就无法忽略。

## 不允许使用异常实现流程控制和条件控制。

  我们可以利用break\continue \if else  配合finally实现流程控制。但利用异常控制流程,比通常意义上的条件语句(if/else、switch)要低效。

## try块放到事务代码中,catch异常后,如果需要回滚事务,一定注意手动回滚事务。

# 参考文献

[1] [详解try-with-resource](http://www.oracle.com/technetwork/cn/articles/java/trywithresources-401775-zhs.html)    
[2] [try-with-resource官方文档](https://docs.oracle.com/javase/tutorial/essential/exceptions/tryResourceClose.html)

原文出处:https://www.cnblogs.com/potato-not-tomato/p/9668728.html   


0人推荐
随时随地看视频
慕课网APP