java基础-异常

女神镇楼

1、概述

    异常是一个在程序执行期间发生的事件,他中断了正在执行的程序的正常指令流。如空指针、数组溢出等。在java中,异常也作为类的实例的形式出现。当某一个方法中发生错误时,这个方法会创建一个对象,并且把他传递给正在运行的系统。这个对象就是异常对象。通过异常处理机制,可以将非正常情况下的处理代码和程序的主逻辑分离,即在编写代码主流程的同时在其他地方处理异常。

    代码示例:

public class ExceptionFirstExpression {
    public static void main(String[] args) {
        // >> TODO try 语句中如果发生了异常(Exception),那么程序会跳转到catch语句。
        // >> TODO Java会将异常相关信息封装在一个异常类的实例中,ex是指向这个异常实例的引用
        // >> TODO "处理"最简单的方法,就是调用printStackTrace将异常信息输出到控制台
        // >> TODO catch语句执行完毕,程序会继续向下顺序执行
        try {
            int[] arr = new int[1];
            arr[1] = 9;
        } catch (Exception ex) {

            int abc = 999;
            ex.printStackTrace();
        }

        try {
            String str = "";
            str.substring(9, 10);
        } catch (Exception ex) {
            ex.printStackTrace();
        }

        System.out.println("程序执行结束");

    }
}

 

2、异常捕捉

    为了保证程序有效的运行,需要对程序发生的异常进行相应的处理。如果某个方法抛出异常,即可以在当前方法中进行捕捉,然后进行处理,也可以将异常向上抛出,由方法调用者进行处理。java中异常捕获结构是try、catch、finally三部分组成,其中try语句块存放的是可能发生异常的java语句,catch程序块在try语句块之后,用来激发被捕获的异常,finally是异常处理结构最后的执行部分,无论try语句块如何退出,都会执行finally语句块。

    语法如下:

try{
    //程序代码块
}
catch(Exceptiontype1 e){
    // 对Exceptiontype1的处理
}catch(Exceptiontype2 e){
    // 对Exceptiontype2的处理
}
...
finally{
    //程序块
}

 

   1)try-catch语句块

    代码示例:

public class Take {
    public static void main(String[] args) {
        try{ //try语句中包含可能出现异常的程序代码
            String str = "lili";
            System.out.println(str + "的年龄是:");
            int age = Integer.parseInt("22L");//数据类型转换
            System.out.println(age);
        }catch (Exception e){//catch语句块用来获取异常信息
            e.printStackTrace(); //输出异常性质
        }

        System.out.println("程序结束");
    }
}

 

 

实例分析:

    程序最后还是输出里“程序结束”的提示信息,没有因为异常终止。可能出现异常的代码,使用try-catch处理,当try代码块中的语句出现异常时,程序就会跳转到catch代码块中执行,执行完之后,会继续执行后面的代码,而不会执行try中发生异常的语句后面的代码。使得程序不会因为异常而影响整个程序的执行。

 

public class CallerExceptionAppMain {
    public static void main(String[] args) throws MyException {
        // >> TODO catch 语句是根据异常类型匹配来捕捉相应类型的异常的。
        // >> TODO 如果类型不匹配,catch语句是不会执行的,异常会继续抛出
        // >> TODO 也就是说,catch (Throwable )会捕捉到所有的异常,包括Error,建议最多只捕捉Exception
        // >> TODO 如果catch一个其实并没有被抛出的checked exception,Java程序会报错,因为Java明确的知道这个类型的异常不会发生
        // >> TODO 如果catch一个unchecked exception,Java程序不会报错
        // >> TODO 如果throws一个其实并没有被抛出的checked exception或者unchecked exception,Java程序不会报错
        Caller1 caller1 = new Caller1();
        System.out.println("调用开始");

        caller1.call2Exception();

        System.out.println("调用结束");
    }
}

 

  2)finally语句块

    完整的异常处理语句一定包含finally语句块,无论程序中有无异常发生,并且之间的try-catch无论是否顺利执行完毕,都会执行finally语句。

public class TryCatchFinallyAppMain {

    private static int VAL = 0;

    public static void main(String[] args) {
        System.out.println(withFinally());
        System.out.println(VAL);
    }

    private static int withFinally() {
        int len = 0;
        try {
            String s = null;
//            String s = "abc";
            return s.length();
        } catch (Exception ex) {
            // >> TODO 异常的处理:在有返回值的情况下,返回一个特殊的值,代表情况不对,有异常
            len = -1;
            System.out.println("执行catch里的return语句");
            return len;
        } finally {
            // >> TODO 可以认为finally语句会在方法返回后,后面的方法开始前,会在return语句后
            // >> TODO 无论是因为return结束还是因为异常结束,finally语句都会执行
            System.out.println("执行finally语句");
            // >> TODO finally里最好不要有return语句
//            return -2;

            // >> TODO finally里给return用的变量值赋值没用
//            len = -2;

            VAL = 999;
            System.out.println("finally语句执行完毕");
        }
    }


}

 

3、java中常见的异常

 

4、自定义异常

   用户只需继承Exception类即可自定义异常类。

    自定义异常类的步骤:

    1)创建自定义异常类。

    2)在方法中通过throw关键字抛出异常对象。

    3)如果在当前抛出异常的方法中处理异常,可以使用try-catch关键字捕获并处理,否则就在方法的声明处通过throws指名要抛出的异常,让方法的调用者处理。

    4)在出现异常方法的调用者中捕获并处理。

    

    代码示例:

public class Tran {
    //定义方法,抛出异常
    static int avg(int number1,int number2) throws MyException{
        if(number1 < 0 || number2<0){
            throw new MyException("不可使用负数");
        }
        if(number1 > 100 || number2 > 100){
            throw new MyException("数值太大了");
        }
        return (number1 + number2) /2;
    }

    public static void main(String[] args) {
        try{
            int result = avg(102,105);
        }catch(MyException e){
            System.out.println(e);
        }
    }

    private static class MyException extends Exception{
        public MyException(String ErrorMessaage){
            super(ErrorMessaage);
        }
    }
}

 

5、在方法中抛出异常

  1)使用throws关键字抛出异常

    throws关键字通常被用在声明方法时,用来指定方法可能抛出的异常,多个异常可以使用“,”分割。

   代码示例:

public class Shoot {
    static void pop() throws NegativeArraySizeException{
        //定义方法并抛出NegativeArraySizeException异常
        int[] arr = new int[-3];
    }
    public static void main(String[] args) {
        try {
            pop();
        }catch (NegativeArraySizeException e){
            System.out.println("pop()方法抛出了异常");
        }
    }
}

 

 

  使用throws抛出异常给上级时,如果不想处理,可以继续向上抛出,但最终要有处理该异常的代码。

   

     2)使用throw关键字抛出异常

    throw关键字通常用在方法体中,并抛出一个异常对象。程序在执行到throw时终止,他后面的语句都不执行。通过throw抛出异常后,如果想在上一级代码中捕获并处理异常,需要在抛出异常的方法中使用throws关键字在方法声明时指明要抛出的异常。如果要捕获throw抛出的异常,必须使用try-catch代码块。

    代码示例:

public class MyException extends Throwable {
    String message;
    public MyException(String ErrorMessage){
        message = ErrorMessage;
    }
    public String getMessage(){
        return message;
    }
}

 

public class Captor {
    static int quotint(int x,int y) throws MyException{
        if(y<0){
            System.out.println("除数不能是负数");
        }
        return x/y;
    }

    public static void main(String[] args) {
        try{
            int result = quotint(3,-2);
        }catch (MyException e){
            System.out.println(e.getMessage());
        }catch (ArithmeticException e){
            System.out.println("除数不能为0");
        }catch (Exception e){
            System.out.println("程序发生了其他的异常");
        }
    }
}

 

    实例分析:

    实例中使用了多个catch语句来捕获异常,如果将catch(Exception e) 代码块放在了最前面,将永远也调用不到他后面的代码块,所以catch语句的顺序不可调换。

 

6、运行时异常

  RuntimeException异常时程序运行过程中产生的异常,java类库中每个包都定义了异常类,所有这些类都是Throwable类的子类。Throwable类派生了两个子类,分别是Exception类和Error类,Error类及其子类用来描述java运行系统中的内部错误,以及资源耗尽的错误,这类错误问题比较严重。Exception类称为非致命性类,可以通过捕获处理使程序继续执行。Exception类又根据错误发生的原因,分为RuntimeException异常和除RuntimeException之外的异常。

 

 

7、异常的使用原则

  java强制用户去考虑程序的强健性和安全性,异常处理不应用来控制程序的正常流程,他的主要作用是捕获程序运行时发生的错误,并进行相应的处理,遵循以下原则:

    1)在当前方法声明中使用try-catch方法捕获异常

    2)当一个方法被覆盖时,覆盖他的方法必须抛出相同的异常或者异常子类

    3)如果父类抛出多个异常,则覆盖方法必须抛出那些异常的一个子集,不能抛出新异常。

public class CallerRtExceptionAppMain {
    public static void main(String[] args) {
        Caller1 caller1 = new Caller1();
        System.out.println("调用开始");
        try {
            caller1.call2RTException();
        } catch (MyRuntimeException ex) {
            // >> TODO 错误的演示!不应该使用异常做正常处理逻辑下的跳转
            System.out.println("凌波微步收到!");
        }
        System.out.println("调用结束");
    }
}

 

8、接口中的异常

public class ImplIntfWithEx implements IntfWithEx {
    @Override
    public void danger() throws Exception {
        // >> TODO 接口中声明了抛出异常,实现类中可以抛,也可以不抛。抛的话必须是接口声明的类或其子类
        throw new Exception("");
    }

    @Override
    public void safe() {
        // >> TODO 接口中没有声明抛出异常,实现类中可以抛RuntimeException,也可以不抛。
        // >> TODO 如果抛 checked exception,就会出错
        // >> TODO 可以选择catch住 checked exception,然后将它封在RuntimeException里
        // throw new Exception();
        // throw new RuntimeException();
    }
}

 

posted @ 2020-03-04 16:22  庞某人  阅读(224)  评论(0编辑  收藏  举报