(一)异常处理机制详解

# 前言

  本文主要是对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块里。

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

## 异常处理程序

  异常处理程序必须紧跟在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是一种“异常说明”方式,它属于方法声明的一部分,跟在形式参数列表之后。

void 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)

posted @ 2018-09-18 14:32  土豆不是番薯  阅读(885)  评论(0编辑  收藏  举报