Java异常和错误

2021-08-28  本文已影响0人  淡季的风

一、概述

Java中, 所以的异常都继承自java.lang.Throwable

image.png

二、异常(Exception)

1、检查异常(CheckedException)

程序运行中,很容易出现的、情理可容的异常。除了RuntimeException及其子类外, 其他的都是检出异常。检出异常必须被处理, Java编译器会检查这种异常, 当遇到这种异常时,要么使用try/catch捕获它, 要么使用throw抛出它。

常见的检查异常:

序号 异常名称 异常描述
1 NoSuchFieldException 表示该类没有指定名称抛出来的异常
2 NoSuchMethodException 表示该类没有指定方法抛出来的异常
3 IllegalAccessException 不允许访问某个类的异常
4 ClassNotFoundException 类没有找到抛出异常
5 IOException IO错误

2、非检查异常 (UnCheckedException)

包括运行时异常RuntimeExcptionError以及其子类,表示编译器不会检查程序是否对其做了处理, 出现运行时异常时,表示程序出现了错误, 需要找出异常并修正它。

序号 异常名称 异常描述
1 ArrayIndexOutOfBoundsException 数组越界异常
2 NullPointerException 空指针异常
3 IllegalArgumentException 非法参数异常
4 NegativeArraySizeException 数组长度为负异常
5 IllegalStateException 非法状态异常
6 ClassCastException 类型转换异常

三、异常处理

1、异常处理的关键字

2、try-with-resources

Java中很多类库都需要在使用完成或异常发生时关闭资源, 比如StreamConnection等等, 传统的处理方式:

     // 打开资源
     Connection conn = //
     try{
         //  使用资源
     }catch(Exception e){
         //  处理异常
     }finally{
          // 判断资源是否被关闭, 如果没有被关闭,则主动关闭资源
          if(conn != null){
               // 关闭资源也可能抛出异常, 需要注意
               conn.close();
          }
     }

凡是继承自java.io.Closeablejava.io.AutoCloseable的子类都可以使用try-with-resources方式进行关闭资源:

try(Connection conn = //){ // 打开资源并且在代码执行结束的时候,无论是否发生异常, 主动关闭资源
   // 使用资源
}catch{
   // 处理异常
}

3、异常处理流程

Q:如果在try/catch中包含中包含return语句, finally中的代码是否会执行?

W: 一定会执行,finally语句中的代码在return前先执行然后再return

Q: 如果try/catchfinally中同时包含return语句会返回那一个?

W: 返回finally语句中返回的内容。finally中包含return语句时, 会使try/catch中的return失效, 最终返回finally中返回的内容.

public class UserClient2 {
    public static void main(String[] args) {
        System.out.println(test());
    }

    public static String test(){
        try{
            throw new Exception("-1"); 
            // return "1";
        }catch (Exception e){
            return "2";
        }finally {
            return "3";
        }
    }
}

// Output
// 无论是否抛出异常, 返回的都是3。

Q: finally中包含对try/catch返回内容的更新操作, 返回值时try/catch返回内容, 还是finally更新后的内容。

W: 如果返回的内容时基本类型,则返回的是try/catch返回的值,如果返回的是引用类型, 则返回的是finally语句更新后的值。

int i = 0;
try{
    i ++ ;
   System.out.println(i);
    return i;  // 1
}catch(Exception e){
    i ++;    //  1
    System.out.println(i);
}finally{
    i++;   // 2
    System.out.println(i);
}

// Output
// 1
// 2
// 最终返回的是1

可以看到,最终返回的是1, 是因为return时,会先执行return前的代码,然后暂存return的返回值, 再执行finally的代码,最后通过return返回之前保存的内容。所以这里返回的是1,而不是2, 但是对于下面这个例子,则不一样:

List<Integer> valueList = new ArrayList();
try{
   valueList.add(1);
   System.out.println(valueList);
    return valueList;  
}catch(Exception e){
    valueList.add(2);
    System.out.println(valueList);
}finally{
    valueList.add(3);
    System.out.println(valueList);
}
// Output
// {1,}
// {1,3}
// 最终输出是{1,3}

最终的输出是{1,3}, 主要是因为返回的数据类型, 当返回的是基本类型时,暂存的内容不会改变, 但是返回的是引用类型时,引用类型存的不是变量本身, 而是对变量的引用, 所以当finllay改变了变量的内容时,最终返回结果也会被改变。

总结:
1、 finally的内容一定会被执行。
2、 当try/catch中包含return时, finally的代码也会被执行。return的时候需要注意返回的值类型, 否则返回值会受到影响。
3、 finally中包含return 时, 会直接返回finaly中的值, 导致try/catch中的return失效。

四、总结

上一篇 下一篇

猜你喜欢

热点阅读