《java JDK7 学习笔记》之异常处理

时间:2023-03-08 16:20:56

1、java中所有的错误都会被打包为对象,JVM会尝试执行try区块中的程序代码,如果发生错误,执行流程会跳离错误发生点,然后比较catch括号中声明的异常类型,是否符合被抛出的错误对象类型,如果是的话,就执行catch区块中的程序代码。

2、错误对象都继承自java.lang.Throwable类,Throwable定义了取得错误信息、堆栈追踪等方法,它有两个子类:java.lang.Error与java.lang.Exception。

3、Error与其子类实例代表严重系统错误,Error对象抛出时,基本上不用处理,任其传播至JVM为止,或者是最多留下日志信息。

4、通常程序中会使用try、catch加以处理的错误,都是Exception或其子类实例,所以通常称错误处理为异常处理,对于某些异常,可以try、catch语法尝试捕捉将应用程序恢复至可执行状态。

5、如果某个方法声明会抛出Throwable、Exception或子类实例,但又不属于java.lang.RuntimeException或其子类实例,就必须明确使用try、catch语法加以处理,或者用throws声明这个方法会抛出异常,否则会编译失败。

6、Throwable、Exception或其子对象,但非属于RuntimeException或其子对象,称为受检异常。

7、属于RuntimeException衍生出来的类实例,通常是事先无法预测错误是否发生的执行时期异常,编译程序不会强迫一定得在语法上加以处理,称为非受检异常。

8、如果父类异常对象在子类异常对象前被捕捉,则catch子类异常对象的区块将永远不会被执行,编译程序会检查出这个错误。

9、从JDK7开始,可以使用多重捕捉语法,不过仍得注意异常的继承,catch括号中,左边的异常不能是右边异常的父类,否则会发生编译错误。

10、操作对象的过程中如果会抛出受检异常,但目前环境信息不足以处理异常,所以无法使用try、catch处理时,可由方法的客户端依据当时调用的环境信息进行处理。为了告诉编译程序这个事实,必须使用throws声明此方法会抛出的异常类型或父类型,编译程序才会允许通过编译。

11、如果是非受检异常,原本就可以自行选择是否处理异常,因此不使用try、catch处理时也不用特别在方法上使用throws声明,不处理非受检异常时,异常会自动往外传播。

12、在catch区块中进行完部分错误处理之后,可以使用throw(注意不是throws)将异常再抛出。

13、在JDK7中,编译程序对于重新抛出的异常类型可以更精准判断。

14、若想得知异常发生的根源,以及多重方法调用下异常的堆栈传播,可以利用异常对象自动收集的堆栈追踪来取得相关信息,例如调用异常对象的printStackTrace()、getStackTrace()等方法。

15、要善用堆栈追踪,前提是程序代码中不可有私吞异常的行为、对异常做了不适当的处理,或显示了不正确的信息。

16、在使用throw重抛异常时,异常的追踪堆栈起点,仍是异常的发生根源,而不是重抛异常的地方。如果想要让异常堆栈起点为重抛异常的地方,可以使用fillInStackTrace(),这个方法会重新装填异常堆栈,将起点设为重抛异常的地方,并返回Throwable对象。

17、无论try区块中有无发生异常,若撰写有finally区块,则finally区块一定会被执行。如果程序撰写的流程中先return了,而且也有finally区块,finally区块会先执行完后,再将值返回。

18、在JDK7之后,新增了尝试关闭资源(try-with-resources)语法,想要尝试自动关闭资源的对象,是撰写在try之后的括号中。

19、若一个异常被catch后的处理过程引发另一个异常,通常会抛出第一个异常作为响应,addSupperssed()方法是JDK7在java.lang.Throwable中新增的方法,可将第二个异常记录在第一个异常之中,JDK7中与之对应的是getSupperssed()方法,可返回Throwable[],代表先前被addSupperssed()记录的各个异常对象。

20、JDK7的尝试关闭资源语法可套用的对象,必须操作java.lang.AutoCloseable接口,这是JDK7新增的接口。尝试关闭资源语法也可以同时关闭两个以上的对象资源,只要中间以分号分隔。在try的括号中,越后面撰写的对象资源会越早被关闭。

课后练习选择题:

1、public class Main{

public static void main(String[] args){

try{

int number = Integer.parseInt(args[0]);

System.out.println(number++);

}catch(NumberFormatException ex){

System.out.println(“必须输入数字”);

}

}

}

执行时若没有指定命令行自变量,以下描述正确的是:显示ArrayIndexOutOfBoundException堆栈追踪。

2、Object[] objs = {“java”,”7”};

Integer number = (Integer) objs[1];

System.out.println(number);

以下描述正确的是:显示ClassCastException堆栈追踪

3、public class Main{

public static void main(String[] args){

try{

int number = Integer.parseInt(args[0]);

System.out.println(number++);

}catch(NumberFormatException ex){

System.out.println(“必须输入数字”);

}

}

}

执行时若指定命令行自变量one,以下描述正确的是: 显示的必须输入数字。

4、public class FileUtil{

public static String readFile(String name) throws                {
                   FileInputStream input = new FileInputStream(name);
                   return null;
           }

}

方法名的throws关键字后要加FileNotFoundException ,声明该方法需要抛出找不到文件的异常。

5、public static String readFile(String name) {
        try {
            FileInputStream input = new FileInputStream(name);
        } catch (              e) {
            e.printStackTrace();
        }
        return null;
    }

catch括号中描述正确的是:填入Throwable或者FileNotFoundException可以通过编译。

6、public class ExtendsDemo1 extends Resourse{
    @Override
    void doService() throws              {}
    
    public static void main(String[] args){

}   
}

class Resourse{
    void doService() throws IOException{}
}

红色区域填入Error、IOException、FileNotFoundException选项都可以通过编译。

7、public class Main{

public static void main(String[] args){

try{

int number = Integer.parseInt(args[0]);

System.out.println(number++);

}catch(ArrayIndexOutOfBouondException  | NumberFormatException ex){

System.out.println(“必须输入数字”);

}

}

}

以下描述正确的是:显示”必须输入数字“。

8、public static void main(String[] args){

try{

int number = Integer.parseInt(args[0]);

System.out.println(number++);

}catch(RuntimeException  | NumberFormatException ex){

System.out.println("必须输入数字");

}

}

以下描述正确的是:编译错误。

注:因为左侧Runtime是右侧NumberFormatException的父类。

9、public static String readFile(String name){
            try( FileInputStream input = new FileInputStream(name)){
               
            }
         }

以下描述正确的是:编译失败、

10、try(ResourceSome some = new  ResourceSome();

RsourceOther other = new ResouceOther()

以下描述正确的是:执行完try后先会关闭ResouceOther。