12.异常

1. 异常的基础知识

  • 异常指程序运行中出现的不期而至的各种情况,如:文件找不到,网络连接失败,非法参数等。

  • 异常发生在程序运行期间,它影响了正常的程序流程。

  • 常见的异常:

package com.exception;

/**
 * @author 23862
 */
public class Demo01 {

    public static void main(String[] args) {

        System.out.println();
        /*报错:Exception in thread "main" java.lang.ArithmeticException: / by zero
        System.out.println(11/0);*/
    }

/*调用时报错:Exception in thread "main" java.lang.StackOverflowError
    public void a(){
        b();
    }
    public void b(){
        a();
    }*/
}
  • 异常的简单分类:

    • 检查性异常:最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在的文件时,异常就发生了,这些异常在编译时不能被简单地忽略。
    • 运行时异常:运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略。
    • 错误ERROR:错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译时是检查不到的。
  • 异常体系结构:

    • Java把异常当做对象来处理,并定义一个基类java.lang.Throwable作为所有异常的超类。
    • 在Java API中已经定义了许多异常类,这些异常类分为两大类,错误Error和异常Exception

在这里插入图片描述

  • Error

    • Error类对象由Java虚拟机生成并抛出,大多数错误与代码编写者所执行的操作无关,对于error异常,Java程序是无能为力的。
    • Java虚拟机运行错误(Virtual MachineError),当JVM不再有继续执行操作所需的内存资源时,将出现OutOfMemoryError。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止;
    • 还有发生在虚拟机试图执行应用时,如类定义错误(NoClassDefFoundError),连接错误(LinkageError)。这些错误是不可查的,因为它们在应用程序的控制和处理能力之外,而且绝大多数是程序运行时不允许出现的状况。
  • Exception

    • Exception分支中有一个重要的子类RuntimeException(运行时异常)

      • ArrayIndexOutOfBoundsException(数组下标越界异常)
  • NullPointerException(空指针异常)

    • ArithmeticException(算数异常)
  • MissingResourceException(丢失资源)

    • ClassNotFoundException(找不到类)等异常,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。
  • 这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生;

    • 非RuntimeException:编译期间就必须处理,否则程序不能通过编译,更不能正常运行
  • Error和Exception的区别:Error通常是灾难性的致命错误,是程序无法控制和处理的,当出现这些异常时,Java虚拟机(Jvm)一般会选择终止线程;Exception通常情况下是可以被程序处理的,并且在程序中应该尽可能的去处理这些异常。

2. 异常处理机制

2.1JVM的默认处理方案

如果程序出现了问题,我们没有做任何处理,最终会做默认的处理:

  • 把异常的名称,异常原因及异常出现的位置等信息输出在了控制台
  • 程序在在出现异常的位置停止执行

2.2异常处理方式

我们自己处理异常有两种方法:

  • 抛出异常(throws)
  • 捕获异常(try…catch…)
  1. try…catch…方式
  • try…catch…格式:
	try {
        可能出现异常的代码;
    }catch(异常类名 变量名) {
        异常的处理代码;
    }

执行流程:程序从try内的代码开始执行,当出现异常时会自动生成一个异常对象提交给Java运行时系统,当Java运行时系统接收到异常对象时回到catch内的代码中寻找匹配的异常类,找到后进行相应处理,当流程执行完毕后,程序还会继续往下执行。

  • 异常处理五个关键字
    • try:监控区域
    • catch:捕获异常
    • finally:处理善后操作
    • throw:在方法内捕获异常
    • throws:在方法上捕获异常
public class ExceptionDemo {
    public static void main(String[] args) {
        System.out.println("开始");//开始
        method();//你访问的数组的索引不存在
        System.out.println("结束");//结束
    }
    
    public static void method() {
        try {
            int[] arr = {1,2,3};
            //数组下标越界
            System.out.println(arr[3]);//出现异常后创建对象new ArrayIndexOutOfBoundsException();
        } catch (ArrayIndexOutOfBoundsException e) {//将创建的对象与catch代码块中的内容相匹配
            System.out.println("你访问的数组的索引不存在");
            e.printStackTrace();//该代码会输出JVM的默认处理方案
        }
    }
}
  • 程序出现异常,JVM也完成默认的处理,而且还正常的执行了异常发生之后的代码,进行try…catch操作最主要的目的就是为了让程序出现异常后能正常执行下去。
  1. throws方式
  • 不是所有异常我们都有权限处理,当遇到无法处理的异常时,需要使用Java提供的throws处理方案

  • throws格式:

  • throws 异常类名;
    
  • 该格式是跟在方法的括号后面的

  • 编译时异常必须要进行处理,两种处理方案:try…catch…或者throws,如果抛出异常,将来谁调用有异常的方法谁就需要去处理异常。

  • 运行时异常可以不处理,出现问题后,需要再次修改代码

  • 测试类:

public class ExceptionDemo {
    public static void main(String[] args) {
        System.out.println("开始");//开始
        method();
        System.out.println("结束");
    }
    
    //运行时异常
    public static void method() throws ArrayIndexOutOfBoundsException {
        int[] arr = {1, 2, 3};
        System.out.println(arr[3]);
    }
}
  • 控制台输出“开始”,异常的类名,原因和位置。“结束”没有被输出。说明用抛出异常方式处理运行时异常不是真正的处理,如果想让程序继续往下执行,还是需要try…catch…去处理。
public class ExceptionDemo {
    public static void main(String[] args) {
        System.out.println("开始");//开始
        try{
            method(); 
        } catch (ParseException e){
            e.printStackTrace();
        }
        System.out.println("结束");//结束
    }
    
    //编译时异常
    public static void method() throws ParseException {
        String s = "2022-05-09";
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
        Date d = sdf.parse(s);
        System.out.println(d);
    }
}
  • 控制台输出"开始",“Sun Aug o9 00:00:00 CST 2022”,“结束”。程序发生异常后成功的往下执行了。

3. Throwable的成员方法

方法名说明
public String getMessage()返回此throwable的详细消息字符串
public String toString()返回此可抛出的简短描述
public void printStackTrace()把异常的错误信息输出在控制台
  • public String getMessage()方法:
public class ExceptionDemo {
    public static void main(String[] args) {
        System.out.println("开始");//开始
        method();
        System.out.println("结束");//结束
    }
    
    public static void method() {
        try {
            int[] arr = {1,2,3};
            //数组下标越界
            System.out.println(arr[3]);//出现异常后创建对象new ArrayIndexOutOfBoundsException();
        } catch (ArrayIndexOutOfBoundsException e) {//将创建的对象与catch代码块中的内容相匹配
            System.out.println(e.getMessage());//方法返回出现异常的原因,Index 3 out bounds for length 3    
        }
    }
}
  • public String toString()方法:
public class ExceptionDemo {
    public static void main(String[] args) {
        System.out.println("开始");//开始
        method();
        System.out.println("结束");//结束
    }
    
    public static void method() {
        try {
            int[] arr = {1,2,3};
            //数组下标越界
            System.out.println(arr[3]);//出现异常后创建对象new ArrayIndexOutOfBoundsException();
        } catch (ArrayIndexOutOfBoundsException e) {//将创建的对象与catch代码块中的内容相匹配
            System.out.println(e.toString());//方法返回出现异常的原因和异常类名,包含了getMessage方法输出的内容。输出:java.lang.ArrayIndexOutOfBoundsException: Index 3 out of bounds for length 3    
        }
    }
}
  • public void printStackTrace()方法:
public class ExceptionDemo {
    public static void main(String[] args) {
        System.out.println("开始");//开始
        method();//你访问的数组的索引不存在
        System.out.println("结束");//结束
    }
    
    public static void method() {
        try {
            int[] arr = {1,2,3};
            //数组下标越界
            System.out.println(arr[3]);//出现异常后创建对象new ArrayIndexOutOfBoundsException();
        } catch (ArrayIndexOutOfBoundsException e) {//将创建的对象与catch代码块中的内容相匹配
            System.out.println("你访问的数组的索引不存在");
            e.printStackTrace();//该代码会输出JVM的默认处理方案在控制台
        }
    }
}

4. 编译时异常和运行时异常的区别

Java中的异常被分为两大类:编译时异常运行时异常,也被称为受检异常非受检异常

所有的RumtimeException类及其子类被称为运行时异常,其他的异常都是编译时异常

  • 编译时异常:必须显示处理(在编写代码时就要处理),否则程序会发生错误,无法通过编译
  • 运行时异常:无需显示处理,可以和编译时异常一样处理,一般通过try…catch…处理
public class ExceptionDemo {
    public static void main(String[] args) {
		method();
    }
    
    //运行时异常
    public static void method() {
        try {
            int[] arr = {1, 2, 3};
        	System.out.println(arr[3]);//在控制台报错ArrayIndexOutOfBoundsException
        }catch (ArrayIndexOutOfBoundsException e) {
            e.printStackTrace();
        }
    }
}
public class ExceptionDemo {
    public static void main(String[] args) {
		method();
    }
    
    //编译时异常
    public static void method() {
        try {
            String s = "2022-05-08";
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
            Date d = sdf.parse(s);//因为parse方法可能会出问题所以必须进行处理,捕获和抛出异常都可以
        	System.out.println(d);
        }catch (ParseException e) {
            e.printStackTrace();
        }
    }
    
}

5. 自定义异常

5.1 使用实例

  • 使用Java内置的异常类可以描述在编程时出现的大部分异常情况。除此之外,用户还可以自定义异常。用户自定义异常类,只需要继承Exception类即可。

  • 格式:

  • public class 异常类名 extends Exception {
        无参构造
        带参构造
    }
    
  • 范例:

  • public class ScoreException extends Exception {
        
        public ScoreException() {}
        
        public ScoreException(String message) {
            super(message);
        }
    }
    
  • 在程序中使用自定义异常类,大体可分为一下几个步骤:

    1. 创建自定义异常类。
    2. 在方法中通过throw关键字抛出异常对象。
    3. 如果在当前抛出异常的方法中处理异常,可以使用try-catch语句捕获并处理;否则在方法的声明处通过throws关键字指明要抛出给方法调用者的异常,继续进行下一步操作。
    4. 在出现异常方法的调用者中捕获并处理异常。
  • 自定义异常类ScoreException:

public class ScoreException extends Exception {
    
    public ScoreException() {}
    
    public ScoreException(String message) {
        super(message);
    }
}
  • Teacher类:
public class Teacher {
    
    public void checkScore(int score) throws ScoreException {//抛出异常类
        if(score<0 || score>100) {
            throw new ScoreException("你给的分数错误,分数应在0到100之间");//抛出自定义的异常对象
        } else {
            System.out.println("分数正常");
        }
    }
}
  • 测试类TeacherTest:
public class TeacherTest {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        System.out.println("请输入分数:");
        int score = sc.nextInt();
        
        Teacher t = new Teacher();
        try{
           t.checkScore(score); //处理编译时异常
        } catch (ScoreException e) {
            e.printStackTrace();
        }
        
    }
}
  • 在控制台输入"90",控制台输出"分数正常"

  • 在控制台输入"111",控制台输出异常的名称和出现异常的位置,以及异常的原因"你给的分数错误,分数应在0到100之间"

  • 实际应用中的经验总结

    • 处理运行异常时,采用逻辑去合理规避同时辅助try-catch进行处理。
    • 在多重catch块后面,可以加上一个catch(Exception)来处理可能会被遗漏的异常。
    • 对于不确定的代码,也可以加上try-catch,处理潜在的异常。
    • 尽量去处理异常,切忌只是简单地调用printStackTrace()去打印输出。
    • 具体如何处理异常,要根据不同的业务需求和异常类型去决定。
    • 尽量添加finally语句块去释放占用的资源。

5.2 throws和throw的区别

  • throws

    • 用在方法声明后面,跟的是异常类名
    • 表示抛出异常,由该方法的调用者来处理
    • 表示出现异常的一种可能性,并不一定会发生这些异常
  • throw

    • 用在方法体内,跟的是异常对象名
    • 表示抛出异常,由方法体内的语句处理
    • 执行throw一定抛出了某种异常
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值