1

概述

说起异常,我就想起了Bug,也就是常说的信春哥,无Bug,什么是Bug呢?我理解的Bug就是没有按照自己原先假想的逻辑去执行,这其中包括了两个方面,一方面是代码语法问题,一方面是逻辑问题,就比如正常逻辑买了东西要付款,但是买家买了东西却没有付款,直接拿东西走了,这是逻辑问题,或者是本来数组存储3个元素,你却存了4个,这时候也出现了Bug,程序报错了,这种Bug就是异常。
异常也是Java类的一种,用new创建对象,他们的结构是这样的:

  • Throwable

    • error

    • Exception

恩,就是这样,异常分为错误和异常,感觉有点绕口,但就是这样,像error这种错误是在编译期间就会报出来的,如果你使用的是IDE,所以我们只需要关注Exception异常就可以了。

获取异常和处理异常

获取和处理异常可能就是异常的全部,我们的一个原则就是:

  • 发生异常就得让程序员知道

获取异常

怎么获取异常呢?Java给出了try关键字用来解决,我们只需要将可能出现异常的代码放在try中就可以了,就像这样:

    try{
        ...
    }

java给出了尽量优雅的解决方案来处理异常,它不希望处理异常的代码和本身的业务代码有过多的混合,获取到异常就该处理异常,这时候就应该这样写了:

    try{
        ...
    }catch(Type1 type1){
        ...
    }catch(Type2 type2){
        ...
    }
  • catch中的type代表了在try中可能出现的异常类型

catch如何匹配异常?

  • 按顺序从第一个到最后一个,如果发现异常匹配,就停止匹配,和switch模式不一样

  • 基类包含子类异常,如果第一个是Exception,那么后面的都不用匹配了

那么catch中怎么处理这个异常呢?

  • 不作处理(会出现“吞食”)

  • 打印到控制台

  • 写入日志

  • 继续向上抛

在try块中抛出异常,我们在catch中会匹配到相应的异常类型,这时候我们就会拿到对应的异常对象的引用,我们调用throwalbe的方法用于处理:

public class ExceptionMethods {
    public static void main(String[] args) {
        try {
            throw new Exception("My Exception");
        } catch (Exception e) {
            System.out.println("Caught Exception");
            System.out.println(e.getMessage());
            System.out.println(e.getLocalizedMessage());
            System.out.println(e);
            System.out.println(e.toString());
            e.printStackTrace();
            e.printStackTrace(System.out);
        }
    }
}
  • 我们一般使用e.printStackTrace();打出异常栈信息即可

  • 或者是写入Log,使用logger.error

  • 每个人处理处理异常的方式可能不尽相同

我们除了可以使用try来捕获异常,我们也可以将这个异常抛出去,我们可以把异常比作现在这种制度,地球村出了点问题,村长衡量了一下,觉得这事情自己处理不了,就交给乡长了,这种把问题交给上层或者是Java编程思想中说的跳出当前环境,交给一个更大的环境去处理,这也是Java异常处理的一种思路。

if(t == null)
    throw new NullPointerException();

其实异常最重要的就是异常的名称,现在Java正在扩充异常的种类,但是很多异常可能不是我们想要的,那么我们就可以自定义异常:

class SystemErrException extends Exception{

}

这就是一个异常类,我们继承了Exception,这个SysteErrException就可以用于我们处理异常的时候了,当然我们可以给他加点参数:

class SystemErrException extends Exception{
    public SystemErrException(){ 
    }
    public SystemErrException (String message){
        super(message);
    }
} 

或者是这样:

class SystemErrException extends Exception{
    private int i;
    public SystemErrException(){ 
    }
    public SystemErrException (String message){
        super(message);
    }
    public SystemErrException (String message, int i){
        super(message);
        this.i = i;
    }
} 
  • 这只不过是多加了几个参数,但是这些一般是没什么必要的。

throws

说到这,就得说另一个关键字throws,看清楚,不是throw,而是throws,这可能也是面试新手的时候会问的一个问题,throws是异常中的一个申明,在IO学习中会非常常见,它是一个声明,编译器检查到说你这段代码可能会发生什么异常,你要声明一下,这时候你就要在方法上声明:

public void inputFile() throws IOException{
    ....
}

finally

finally用处就和他的意思相符,表示最终的含义,也就是finally里面的代码一定会得到执行:
案例一

public class FinallyException {
    static int count = 0;

    public static void main(String[] args) {
        while (true){
            try {
                if (count++ == 0){
                    throw new ThreeException();
                }
                System.out.println("no Exception");
            }catch (ThreeException e){
                System.out.println("ThreeException");
            }finally {
                System.out.println("in finally cause");
                if(count == 2)
                    break;
            }
        }
    }
}

class ThreeException extends Exception{

}

执行结果:

ThreeException
in finally cause
no Exception
in finally cause

案例二:return案例

public class MultipleReturns {

    public static void f(int i){
        System.out.println("start.......");
        try {
            System.out.println("1");
            if(i == 1)
                return;
            System.out.println("2");
            if (i == 2)
                return;
            System.out.println("3");
            if(i == 3)
                return;
            System.out.println("else");
            return;
        }finally {
            System.out.println("end");
        }
    }

    public static void main(String[] args) {
        for (int i = 1; i<4; i++){
            f(i);
        }
    }
}

执行结果:

start.......
1
end
start.......
1
2
end
start.......
1
2
3
end

最佳实践

我们可能遇到这种情况,打开一个文件,读取文件,关闭文件,正常逻辑是这样的,但是这样会有几个问题:

  • 在打开文件的过程中出现异常,但是还要关闭文件

就像这样:

    try{
        A a = new A();
        ...
    }catch(){
        ...
    }finally{
        a.close();
    }
  • A在创建过程中会出现异常,但是还是会执行a的方法,比如关闭文件

所以我们应该这么做:

    try{
        A a = new A();
        try{
            ...
        }finally{
            a.close();
        }
    }catch(){
        ...
    }

异常栈

一般报错都会报一大堆错误,大家无从看起,其实异常报错也是有规律的,这就是栈结构,先进后出,举个栗子:

public class WhoCalled {
    static void f() {
        try {
            throw new Exception();
        } catch (Exception e) {
            for (StackTraceElement ste : e.getStackTrace()){
                System.out.println(ste.getMethodName());
            }
        }
    }

    static void g(){
        f();
    }

    static void h(){
        g();
    }


    public static void main(String[] args) {
        f();
        System.out.println("---------------------------");
        g();
        System.out.println("---------------------------");
        h();
        System.out.println("---------------------------");

    }
}

运行结果:

f
main
---------------------------
f
g
main
---------------------------
f
g
h
main
---------------------------

可以看到异常信息都是从内到外的,按我的理解查看异常的时候要从第一条异常信息看起,因为那是异常发生的源头。

总结

异常总的来说还是很重要的,也是保障程序健壮性很重要的一栏,并且可以达到立竿见影的效果,这里只是基本总结了异常的一些常见问题,很多还得在自己运用的过程中去探索。


sailfishc
47 声望9 粉丝

喜欢孤独与自由