美文网首页Java 杂谈
Java入门系列-20-异常

Java入门系列-20-异常

作者: 要成为王的男人 | 来源:发表于2018-10-24 22:25 被阅读25次

    为什么要进行异常处理

    下面这段代码能否正常执行

    public class DemoCalc {
        public static void main(String[] args) {
            int a=0;
            int b=0;
            int c=a/b;
            System.out.println("运算结果为:"+c);
        }
    }
    

    结果是我们在控制台中看到一段错误提示,那是因为除数不能为0。异常就是在程序运行过程中发生的不正常事件,会中断运行的程序

    Java 使用了异常处理机制为程序提供了错误处理的能力,在程序中预先设置好对付异常的处理办法,待程序发生异常时对异常进行处理,处理完毕后,程序便可以继续运行。

    下面来看一下Java中是如何进行异常处理的

    如何进行异常处理

    Java 的异常处理是通过5个关键字实现的:try、catch、finally、throw、throws

    关键字 作用
    try 执行可能产生异常的代码
    catch 捕获异常
    finally 无论是否发生异常,代码总能执行
    throws 声明方法要抛出的异常
    throw 手动抛出异常

    常见异常及异常分类

    Throwable 是Java 中所有错误和异常的父类

    Error类:Throwable的子类,仅靠程序本身无法恢复的严重的错误。

    Exception类:Throwable的子类,由Java应用程序抛出和处理的非严重错误

    RuntimeException类:Exception的子类,运行时异常,不要求程序必须做出处理。

    Checked异常:Exception的子类,程序必须处理该类异常。

    常见异常类型

    异常类型 说明
    Exception 异常层次结构的父类
    ArithmeticException 算数错误情形,如以零作除数
    ArrayIndexOutOfBoundsException 数组下标越界
    NullPointerException 尝试访问null对象成员
    ClassNotFoundException 不能加载所需的类
    IllegalArgumentException 方法接收到非法参数
    ClassCastException 对象强制类型转换出错
    NumberFormatException 数字格式转换异常,如把"abc"转换成数字

    try-catch

    语法:

    public void method(){
        try{
            //代码段1
            //可能产生异常的代码段
            //代码段2
        }catch(异常类型 ex){
            //对异常进行处理的代码段
        }
        //代码段
    }
    

    try-catch块捕获异常有三种情况:

    1、try块中没有任何异常,try中正常,catch不会执行,正常执行try-catch后的代码。

    2、try块中可能发生异常的代码段发生异常,代码段2不会执行,而是执行catch中异常处理的代码,正常执行try-catch后的代码。

    catch中异常类型的printStackTrace() 方法能进行堆栈跟踪显示出程序运行到当前类的执行流程,异常堆栈信息中包含了异常的类型及异常出现的位置。

    3、异常类型不匹配,程序将中断。比如try产生的异常为ArithmeticException,catch却用了 ClassCastException。

    在控制台中接收数字做除法运算

    import java.util.Scanner;
    
    public class DemoInput {
        public static void main(String[] args) {
            Scanner input=new Scanner(System.in);
            try{
                System.out.println("请输入被除数(整数):");
                int a=input.nextInt();
                System.out.println("请输入除数(整数):");
                int b=input.nextInt();
                int c=a/b;
                System.out.println("结果:"+c);
            }catch(Exception ex) {
                ex.printStackTrace();
            }
            System.out.println("程序结束");
        }
    }
    

    try-catch-finally

    语法:

    public void method(){
        try{
            //可能会发生异常的代码
        }catch(异常类型 ex){
            //异常处理
        }finally{
            //无论如何都要执行的代码
        }
    }
    

    finally块:是否发生异常都执行

    finllay块不执行的唯一情况:之前的代码中执行了 System.exit(1); 退出虚拟机

    try-catch-finally的使用

    import java.io.FileNotFoundException;
    import java.util.Scanner;
    
    public class DemoInput {
        public static void main(String[] args) {
            Scanner input=new Scanner(System.in);
            try{
                System.out.println("请输入被除数(整数):");
                int a=input.nextInt();
                System.out.println("请输入除数(整数):");
                int b=input.nextInt();
                int c=a/b;
                System.out.println("结果:"+c);
            }catch(Exception ex) {
                ex.printStackTrace();
            }finally {
                System.out.println("感谢您的使用");
            }
            System.out.println("程序结束");
        }
    }
    

    如果在try块或catch块中有return语句,finally是否还会执行?运行下面代码断点调试观察结果。

    public class TestReturn {
    
        public static void main(String[] args) {
            try {
                int a=1+1;
                System.out.println("try执行");
                return;
            } catch (Exception e) {
                System.out.println("catch执行");
            }finally {
                System.out.println("finally执行");
            }
        }
    }
    

    try块或catch块中可以有return语句,如果有return语句会先执行finally最后再执行return。

    多重catch

    try块中可能会发生多种异常,如果要不同的异常进行不同的处理,需要使用多重catch进行处理。

    语法:

    public void method(){
        try{
            //可能发生异常的代码段
        }catch(异常类型1 e){
            //对异常类型1进行的处理的代码段
        }catch(异常类型2 e){
            //对异常类型2进行的处理的代码段
        }catch(异常类型n e){
            //对异常类型n进行的处理的代码段
        }
    }
    

    当try块中发生异常后,会逐个与catch中的异常类型进行匹配,匹配成功后,进入对应的catch进行异常处理,处理完成后不再进入其他catch,程序继续执行。

    排列catch语句的顺序是:先子类后父类

    发生异常时按顺序逐个匹配

    只执行第一个与异常类型匹配的catch语句

    将之前的代码 DemoInput.java 改造成多重catch

    import java.io.FileNotFoundException;
    import java.util.InputMismatchException;
    import java.util.Scanner;
    
    public class DemoInput {
        public static void main(String[] args) {
            Scanner input=new Scanner(System.in);
            try{
                System.out.println("请输入被除数(整数):");
                int a=input.nextInt();
                System.out.println("请输入除数(整数):");
                int b=input.nextInt();
                int c=a/b;
                System.out.println("结果:"+c);
            }catch(InputMismatchException e) {
                System.out.println("输入的数有误!");
            }catch(ArithmeticException e) {
                System.out.println("除数不能为0");
            }catch(Exception ex) {
                System.out.println("发生未知异常");
            }finally {
                System.out.println("感谢您的使用");
            }
            System.out.println("程序结束");
        }
    }
    

    声明异常 throws

    如果一个方法体内抛出了异常如何通知调用者,可以在方法上声明异常。

    
    public class TestThrows {
        
        //声明异常,多个异常可以用逗号隔开
        public void test()throws Exception,ClassNotFoundException{
            //可能会发生异常的代码
        }
    }
    

    处理方式一:调用者处理异常

    public static void main(String[] args) {
        TestThrows t=new TestThrows();
        try {
            t.test();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    

    处理方式二:调用者继续声明异常

    public static void main(String[] args) throws ClassNotFoundException, Exception {
        TestThrows t=new TestThrows();
        t.test();
    }
    

    main方法继续声明异常,调用者就变成虚拟机了,发生异常则按默认方式处理,打印出来。

    抛出异常 throw

    除了系统自动抛出的异常外,有些问题需要程序员自行抛出异常

    public class TestThrow {
        public void inputAge(int age) throws Exception {
            if (age<1) {
                throw new Exception("还有这种年龄?");
            }
        }
        
        public static void main(String[] args) {
            TestThrow t=new TestThrow();
            try {
                t.inputAge(-1);
            } catch (Exception e) {
                System.out.println("年龄有误:"+e.getMessage());
            }
        }
    }
    

    自行抛出异常后,还需要在方法上声明异常

    相关文章

      网友评论

        本文标题:Java入门系列-20-异常

        本文链接:https://www.haomeiwen.com/subject/ctsftqtx.html