第六章 异常机制

异常处理的三种方式

  • 捕获异常:try-catch-finally
  • 声明异常:throws
  • 自定义异常类只需从Exception类或者它的子类派生一个子类即可。

常见的异常类型

  • 算术异常(ArithmeticException)
  • 空指针异常(NullPointerException)
  • 类型转换异常(ClassCastException)
  • 数组下标越界异常(ArrayIndexOutOfBoundsException)
  • 数字格式化异常(NumberFormatEx0ception) 如字符串转数字时内有非数字字符

6.3.3 RuntimeException运行时异常

派生于RuntimeException的异常,如被 0 除、数组下标越界、空指针等
处理:
系统自动检测并将它们交给缺省的异常处理程序(用户可不必对其处理)

6.3.4 CheckedException已检查异常

所有不是RuntimeException的异常,统称为Checked Exception,又被称为“已检查异常”,如IOException、SQLException等以及用户自定义的Exception异常
处理:
使用“try/catch”捕获异常、使用“throws”声明异常

6.4 异常的处理方式之一:捕获异常

捕获异常是通过3个关键词来实现的:try-catch-finally。用try来执行一段程序,如果出现异常,系统抛出一个异常,可以通过它的类型来捕捉(catch)并处理它,最后一步是通过finally语句为异常处理提供一个统一的出口,finally所指定的代码都要被执行(catch语句可有多条;finally语句最多只能有一条,根据自己的需要可有可无)。
详解:
1. try:
try语句指定了一段代码,该段代码就是异常捕获并处理的范围。在执行过程中,当任意一条语句产生异常时,就会跳过该条语句中后面的代码。代码中可能会产生并抛出一种或几种类型的异常对象,它后面的catch语句要分别对这些异常做相应的处理。
一个try语句必须带有至少一个catch语句块或一个finally语句块 。

注意事项

当异常处理的代码执行结束以后,不会回到try语句去执行尚未执行的代码。
会直接跳转去执行 finally中的代码( 判断出 catch 不会再执行其他catch(父类))

2. catch:

  • 每个try语句块可以伴随一个或多个catch语句,用于处理可能产生的不同类型的异常对象。
  • 常用方法,这些方法均继承自Throwable类 。
  • toString ()方法,显示异常的类名和产生异常的原因
  • getMessage()方法,只显示产生异常的原因,但不显示类名。
  • printStackTrace()方法,用来跟踪异常事件发生时堆栈的内容。
  • catch捕获异常时的捕获顺序:
  • 如果异常类之间有继承关系,在顺序安排上需注意。越是顶层的类,越放在下面,再不然就直接把多余的catch省略掉。 也就是先捕获子类异常再捕获父类异常。

2. finally:

  • 有些语句,不管是否发生了异常,都必须要执行,那么就可以把这样的语句放到finally语句块中。
  • 通常在finally中关闭程序块已打开的资源,比如:关闭文件流、释放数据库连接等。
    try-catch-finally语句块的执行过程:
    try-catch-finally程序块的执行流程以及执行结果比较复杂。

基本执行过程如下
程序首先执行可能发生异常的try语句块。如果try语句没有出现异常则执行完后跳至finally语句块执行;如果try语句出现异常,则中断执行并根据发生的异常类型跳至相应的catch语句块执行处理。catch语句块可以有多个,分别捕获不同类型的异常。catch语句块执行完后程序会继续执行finally语句块。finally语句是可选的,如果有的话,则不管是否发生异常,finally语句都会被执行。

注意事项

  1. 即使try和catch块中存在return语句,finally语句也会执行。是在执行完finally语句后再通过return退出。
  2. finally语句块只有一种情况是不会执行的,那就是在执行finally之前遇到了System.exit(0)结束程序运行。
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
public class Test8 {
    public static void main(String[] args) {
        FileReader reader = null;
        try {
            reader = new FileReader("d:/a.txt");
            char c = (char) reader.read();
            char c2 = (char) reader.read();
            System.out.println("" + c + c2);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            try {
                if (reader != null) {
                    reader.close();
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

6.5 异常的处理方式之二:声明异常(throws子句)

  • 当CheckedException产生时,不一定立刻处理它,可以再把异常throws出去。
  • 在方法中使用try-catch-finally是由这个方法来处理异常。但是在一些情况下,当前方法并不需要处理发生的异常,而是向上传递给调用它的方法处理。
  • 如果一个方法中可能产生某种异常,但是并不能确定如何处理这种异常,则应根据异常规范在方法的首部声明该方法可能抛出的异常。
  • 如果一个方法抛出多个已检查异常,就必须在方法的首部列出所有的异常,之间以逗号隔开。

注意事项:
方法重写中声明异常原则:子类重写父类方法时,如果父类方法有声明异常,那么子类声明的异常范围不能超过父类声明的范围。

import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
 
public class Test9 {
    public static void main(String[] args) {
        try {
            readFile("joke.txt");
        } catch (FileNotFoundException e) {
            System.out.println("所需文件不存在!");
        } catch (IOException e) {
            System.out.println("文件读写错误!");
        }
    }  
    public static void readFile(String fileName) throws FileNotFoundException,          
    IOException {
        FileReader in = new FileReader(fileName);
        int tem = 0;
        try {
            tem = in.read();
            while (tem != -1) {
                System.out.print((char) tem);
                tem = in.read();
            }
        } finally {
            in.close();
        }
    }
}

6.6 自定义异常

1.在程序中,可能会遇到JDK提供的任何标准异常类都无法充分描述清楚我们想要表达的问题,这种情况下可以创建自己的异常类,即自定义异常类。
2.自定义异常类只需从Exception类或者它的子类派生一个子类即可。
3.自定义异常类如果继承Exception类,则为受检查异常,必须对其进行处理;如果不想处理,可以让自定义异常类继承运行时异常RuntimeException类。
4.习惯上,自定义异常类应该包含2个构造器:一个是默认的构造器,另一个是带有详细信息的构造器。
使用异常机制的建议

  1.要避免使用异常处理代替错误处理,这样会降低程序的清晰性,并且效率低下。

  2.处理异常不可以代替简单测试---只在异常情况下使用异常机制。

  3.不要进行小粒度的异常处理---应该将整个任务包装在一个try语句块中。

  4.异常往往在高层处理(先了解!后面做项目会说!) 。
/**IllegalAgeException:非法年龄异常,继承Exception类*/
class IllegalAgeException extends Exception {
    //默认构造器
    public IllegalAgeException() {
     
    }
    //带有详细信息的构造器,信息存储在message中
    public IllegalAgeException(String message) {
        super(message);
    }
}
class Person {
    private String name;
    private int age;
 
    public void setName(String name) {
        this.name = name;
    }
 
    public void setAge(int age) throws IllegalAgeException {
        if (age < 0) {
            throw new IllegalAgeException("人的年龄不应该为负数");
        }
        this.age = age;
    }
 
    public String toString() {
        return "name is " + name + " and age is " + age;
    }
}
 
public class TestMyException {
    public static void main(String[] args) {
        Person p = new Person();
        try {
            p.setName("Lincoln");
            p.setAge(-1);
        } catch (IllegalAgeException e) {
            e.printStackTrace();
            System.exit(-1);
        }
        System.out.println(p);
    }
}

在这里插入图片描述

课后部分习题

1.以下关于异常的代码的执行结果是( )。(选择一项)

public class Test {
    public static void main(String args[]) {
        try {
            System.out.print("try");           
            return;
        } catch(Exception e){
            System.out.print("catch");
        }finally {
            System.out.print("finally");
        }
    }
}

A.try catch finally

B.catch finally

C.try finally

D.try
*3.阅读如下Java代码,其中错误的行是( )。(选择二项)

public class Student {
    private String stuId;
    public void setStuId(String stuId) throw Exception { // 1
        if (stuId.length() != 4) { // 2
            throws new Exception("学号必须为4位!"); // 3
        } else {
            this.stuId = stuId;     //4
        }
    }
}

A.1

B.2

C.3

D.全部正确
//反了

简答题

  1. Error和Exception的区别:
  • error 机器无所处理的异常(编码者写错了)
  • Exception 机器可以直接处理的异常
  1. Checked异常和Runtime异常的区别。
  • Checked异常 这种在编译时 就必须做出处理 否则木有办法通过编译
  • Runtime异常 除了Checked的都是 他需要用到try {}catch{} 或者使用throws 抛出
  1. Java异常处理中,关键字try、catch、finally、throw、throws分别代表什么含义?

1) 用try来执行一段程序
2)如果出现异常,系统抛出一个异常,可以通过它的类型来捕捉(catch)并处理它
3)最后一步是通过finally语句为异常处理提供一个统一的出口,finally所指定的代码都要被执行(catch语句可有多条;finally语句最多只能有一条,根据自己的需要可有可无)。
4)throw指抛出的一个具体的异常类型
5)throws是用来声明一个方法可能抛出的所有异常信息,throws是将异常声明但是不处理,而是将异常往上传,谁调用我就交给谁处理。

————————————————
版权声明:本文为CSDN博主「Jeloys」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_38371360/article/details/90044544

发布了55 篇原创文章 · 获赞 1 · 访问量 961

猜你喜欢

转载自blog.csdn.net/weixin_43556527/article/details/103449150