黑马程序员全套Java教程_Java基础教程_异常(含扩展)(二十三)

1.1 异常概述与异常体系结构

  • 在开发过程中,即是我们把代码写得尽善尽美,在系统运行过程中仍然会遇到一些问题,因为很多问题不是靠代码能够避免的,如:客户输入数据的格式读取文件是否存在网络是否始终保持通畅等
  • 异常:在Java语言中,将程序执行中发生的不正常情况称为“异常”。(开发过程中的语法错误和逻辑错误等代码问题不是异常)
  • Java程序执行时发生的异常事件的分类:
    (1)Error:Java虚拟机无法解决的严重问题(程序执行分为编译和运行两个过程,运行的时候特别需要使用JVM)。如:JVM系统内部错误、资源耗尽(比如StackOverflowError(栈溢出,如下例:递归调用导致的栈溢出)OOM(堆溢出))等严重情况。一般不(无法)编写针对性的代码进行处理。
    (2)Exception(狭义上的异常,我们所说的异常处理通常就是指这个,不包括Error):其他因编程错误或偶然的外在因素导致的一般性问题,可以使用针对性的代码进行处理。如:
    (1)空指针访问
    (2)试图读取不存在的文件
    (3)网络连接中断
    (4)数组角标越界
//资源耗尽的两种情况
//1、递归调用导致的栈溢出java.lang.StackOverflowError
public class ErrorTest {
    public static void main(String[] args) {
        main(args);
    }
}

//2、new的空间过大,堆溢出java.lang.OutOfMemoryError
public class ErrorTest {
    public static void main(String[] args) {
        Integer[] arr = new Integer[1024*1024*1024];
    }
}
  • 异常的两种解决方法:一是遇到错误就终止程序的执行(默认)。二是由程序员在编写程序时,就考虑到错误的检测、错误消息的提示,以及错误的处理。
  • 异常体系结构:
    在这里插入图片描述
    Error和Exception是两个类,我们可以查看api,它们的父类是Throwable(为顶级父类)。
    (1)java.lang.Error:An Error is a subclass of Throwable that indicates serious problems that a reasonable application should not try to catch. Most such errors are abnormal conditions. The ThreadDeath error, though a “normal” condition, is also a subclass of Error because most applications should not try to catch it.
    Error是Throwable的一个子类,它表示合理的应用程序不应该捕捉的严重问题。大多数这样的错误都是异常情况。
    除了ThreadDeath,其他错误基本都是以*****Error的格式命名(如java.lang.StackOverflowError和java.lang.OutOfMemoryError),ThreadDeath错误虽然是“正常”情况,但也是错误的一个子类,因为大多数应用程序不应该尝试捕捉它。
    综上,对于Error,我们一般不编写针对性的代码进行处理。
    (2)java.lang.Exception:上图Exception类下,红色为编译时异常,蓝色为运行时异常,关于两者区别可转到本文1.6。

1.2 JVM遇到异常时的默认处理方案

运行下列程序:

public class ExceptionDemo2 {
    public static void main(String[] args) {
        System.out.println("开始");
        method();
        System.out.println("结束");
    }

    public static void method() {
        int[] arr = {1, 2, 3};
        System.out.println(3);
    }
}

控制台会输出结果:

开始
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 3
	at itheima.ExceptionDemo2.method(ExceptionDemo2.java:18)
	at itheima.ExceptionDemo2.main(ExceptionDemo2.java:12)

可以看到在输出“开始之后”,控制台在第二行输出了异常的类名ArrayIndexOutOfBoundsException及原因,第三行输出了异常出现的位置(第18行)。而程序并没有输出“结束”,这说明JVM将异常的对应信息输出之后,就将程序结束了。

  • 综上,如果程序出现了问题,我们没有做任何处理,最终会作默认的处理:
    (1)把异常的名称、异常原因以及异常出现的位置等信息输出在了控制台;
    (2)程序停止执行。

1.3 异常处理

  • 如果程序出现了问题,我们需要自己来处理,有两种方案:
    (1)try……catch……
    (2)throws

1.4 异常处理之try……catch……

  • 格式:
try{
	可能出现异常的代码
} catch(异常类名 变量名) {
	异常的处理代码;
}
  • 执行流程:程序从try里面的代码开始执行。出现异常,会自动生成(new)一个异常类对象,该异常对象将被提交给Java运行时系统。当Java运行时系统接收到异常对象时,会到catch中去找匹配的异常类,找到后进行异常的处理。执行完毕之后,程序还可以继续往下执行。
public class ExceptionDemo2 {
    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){
            //System.out.println("你访问的数组对象不存在");
            e.printStackTrace();
        }

    }
}

1.5 Throwable的成员方法

方法名说明
public String getMessage()返回Throwable的详细消息字符串
public String toString()返回可抛出的简短描述
public void printStackTrace()把异常信息错误输出在控制台
    public static void method() {
        try {
            int[] arr = {1, 2, 3};
            System.out.println(arr[3]);
        } catch (ArrayIndexOutOfBoundsException e){//ArrayIndexOutOfBoundsException
            System.out.println(e.getMessage());//3
            System.out.println(e.toString());//java.lang.ArrayIndexOutOfBoundsException: 3
        }

    }

    public static void method2(){
        try{
            System.out.println(1000/0);
        }catch (Exception e){//ArithmeticException
            System.out.println(e.getMessage());//输出/ by zero
            System.out.println(e.toString());//java.lang.ArithmeticException: / by zero
        }
    }

getMessage():返回Throwable的详细消息字符串(视JDK版本有所不同),我们可以通过ctrl+B查看Throwable类下的getMessage(),这个方法返回了String类型的成员变量detailMessage,并在这个方法往上面翻,我们会发现很多Throwable的构造方法,这些构造方法对detailMessage进行了赋值。可知在我们new某种异常对象的时候,子类根据传入的参数(包括无参和带参)调用对应的Throwable父类参构造方法,在构造方法中对detailMessage进行赋值,我们通过getMessage()就可以得到这个值。

public class Throwable implements Serializable {
    private String detailMessage;
	略。。。
	public Throwable(String message) {
        构造方法实现过程。。。
        detailMessage = message;
    }
    其他构造法。。。
    public String getMessage() {
        return detailMessage;
    }
    略。。。
    
}

toString():返回可抛出的简短描述,包括异常的原因(即getMessage()的内容)及异常的类名。
printStackTrace():把异常信息错误输出在控制台,包括异常的类名、原因及位置信息。此方法输出信息比较全,所以我们一般调用这个方法查看异常的相关信息。

1.6 编译时异常和运行时异常的区别

  • 异常的体系结构:捕获异常(异常也是一个对象,捕获即catch)最理想的是在编译期间(javac.exe执行时),但有的错误只有在运行时(java.exe执行时)才会发生(比如:除数为0数组下表越界等)。据此,我们将异常分为编译时异常运行时异常
    (1)运行时异常(也称非受检异常):指编译器不要求强制处置的异常。一般指编程时的逻辑错误,是程序员应该积极避免其出现的异常。包含java.lang.RuntimeException类及它的子类。对于这类异常,可以不作处理,因为这类异常很普遍,若全部处理可能会对程序的可读性和运行效率产生影响。(无需显示处理,也可以和编译时异常一样处理)
    (2)编译时异常(也称受检异常):是指编译器要求必须处置的异常。即程序在运行时由于外界因素造成的一般性异常。编译器要求Java程序必须捕获或声明所有编译时异常。对于这类异常,如程序不处理,可能会带来意想不到的结果。(必须显示处理,否则程序就会发生错误,无法通过编译)
    (3)tip:可以通过在API文档内搜索对应的异常类查看其父类及祖宗类是否有RuntimeException,有的话说明是编译时异常,否则即为运行时异常。
    在这里插入图片描述
public class ExceptionDemo3 {
    public static void main(String[] args) {
        System.out.println("开始");
        method2();
        System.out.println("结束");
    }

    //编译时异常
    public static void method2() {
        String s = "2021-11-11";
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
        //报红:Unhandled exception: java.text.ParseException(Unhandled未经手触过的,未经处理的)
        //通过查看API文档,知其父类为Exception类,说明为编译时异常
        //Date date = sdf.parse(s);
        //System.out.println(date);

        //解决方案
        try{
            Date date = sdf.parse(s);
            System.out.println(date);//Thu Nov 11 00:00:00 CST 2021
        }catch (ParseException e){
            e.printStackTrace();
        }
    }

    //运行时异常
    public static void method() {
        int[] arr = {1, 2, 3};
        //ArrayIndexOutOfBoundsException,其爷爷类为java.lang.RuntimeException,说明为运行时异常
        System.out.println(arr[3]);

        //解决方案
        //try{
        在这里我们可以知道,编译时异常并不是一定会出现的异常,只是有可能出现,只要我们字符串s与sdf的格式是匹配的,就不会出现问题
        而编译器是知道会出现问题,所以报红告诉我们,让我们解决
        //    System.out.println(arr[3]);
        //}catch (ArrayIndexOutOfBoundsException e){
        //    e.printStackTrace();
        //}
    }
}

1.7 异常处理之throws

  • 虽然我们通过try…catch…可以对异常进行出路,但是并不是所有的情况我们都有权限进行异常的处理。也就是说,有些时候可能出现的异常是我们解决不了的,这个时候怎么办呢?Java对此提供了throws的处理方案
  • 格式(可以在异常报红处Alt+Enter快速生成):
throws 异常类名;

注意:这个格式是跟在方法后面的

public class ExceptionDemo4 {
    public static void main(String[] args) {
        System.out.println("开始");

        //method();

        //抛出的异常没有进行处理,现在调用的时候还是要处理
        try{
            method2();
        }catch (Exception e){
            e.printStackTrace();
        }

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

    //编译时异常
    //先不对异常进行处理,只是抛出(延迟)异常
    public static void method2() throws ParseException {
        String s = "2021-11-11";
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
        Date date = sdf.parse(s);
        System.out.println(date);//Thu Nov 11 00:00:00 CST 2021
    }

    //运行时异常
    public static void method() throws ArrayIndexOutOfBoundsException {
        int[] arr = {1, 2, 3};
        System.out.println(arr[3]);//只输出了“开始”没有输出“结束”
    }
}

不管是运行时异常还是编译时异常,我们都可以在方法后面通过throws抛出,但是这个抛出并没有做实际的处理,真正的处理还是得通过try…catch…实现。也就是说,只要你想让程序继续往下执行,就要使用try…catch…进行处理,只不过假如我们处理不了异常,我们可以抛出去(即延迟处理),在调用的时候再进行处理(让调用者处理)。
编译时异常必须要进行处理,两种处理方案:try…catch…或者throws,如果采用throws这种方案,将来谁调用谁处理。
运行时异常可以不处理,出问题后,需要我们回来修改代码。

1.8 自定义异常

  • 虽然Java提供了很多异常类供我们使用,但是在实际开发中,这里类并不能满足我们所有的需求。比如说我们想设置学生的考试成绩只能在0~100之间。所以就需要我们自己定义异常类来实现需求。
  • 那么我们如何让自己定义的异常类称为异常体系的一员呢?
    在IDEA界面按快捷键Ctrl+N,搜索NullPointException类,发现其继承自RuntimeException类;再搜索ParseException类,发现其继承自Exception类。
    在这里插入图片描述
    在这里插入图片描述
    也就是说如果我们自定义的类继承自RuntimeException和Exception,类就可以作为异常体系的一员。
    格式:
public class 异常类名 extends Exception(){
	无参构造
	带参构造
}

范例:

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

为什么带参构造方法要将message传给父类Exception类呢?我们Ctrl+B查看super的构造方法:

    public Exception(String message) {
        super(message);
    }

Ctrl+B继续跟进super

    public Throwable(String message) {
        fillInStackTrace();
        detailMessage = message;
    }

这里的message传给了detailMessage,在上文1.5Throwable的成员方法学习中我们了解到,detailMessage是Throwable的一个成员变量,我们把message传给父类之后,detailMessage有了值,将来就可以通过getMessage()或者printStackTrace()来查看detailMessage。

public class ScoreException extends Exception{
    public ScoreException(){}

    public ScoreException(String message){
        super(message);
    }
}
public class Teacher {

    public void check(int score) throws ScoreException {
        //不符合分数范围的条件,所以我们就要产生一个异常对象并抛出
        if(score < 0 || score>100){
            //抛出异常对象的关键字是throw(后面没有“s”,用于在方法体内部抛出对象)
            //自定义异常类的异常对象需要我们手动throw,而Java提供的异常类如果我们不抛出他也会自动new一个异常对象并throw
            //因为抛出的ScoreException异常继承自Exception,为编译时异常,所以check()要用throws将异常类(是类,不是对象)抛出,将来在调用check()的时候就会报红,我们就可以知道这个方法有一个编译时异常需要进行处理
            //当然,如果ScoreException继承自RuntimeException,可以throws也可以不throws
            //无参构造:不显示异常产生原因
            //throw new ScoreException();

            //带参构造方法:显示异常产生原因
            throw new ScoreException("你输入的分数不合法");
        }else{
            System.out.println("分数合法");
        }
    }
}
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.check(score);
        } catch (ScoreException e) {
            e.printStackTrace();
        }
    }
}

运行结果:
在这里插入图片描述

throws和throw的区别:
(1)throws用在方法声明后面,跟的是异常类名;而throw用在方法体内,跟的是异常对象名;
(2)throws表示抛出异常,由该方法的调用者来处理;而throw表示抛出异常,由方法体内的语句处理;
(3)throws表示出现异常的一种可能性,并不一定会发生这些异常;而执行throw一定抛出了某种异常。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值