Java——异常

文章介绍了Java中的异常概念,包括异常的顶层类Throwable及其子类Error和Exception。异常分为受查异常(编译时异常)和非受查异常(运行时异常)。异常处理涉及try-catch-finally关键字和throws声明。还讨论了如何自定义异常类,并提供了一个登录场景的异常处理示例。
摘要由CSDN通过智能技术生成

一、异常的概念

在Java中,将执行过程中发生的不正常行为称之为异常。


二、异常的体系结构

1.Throwable:是异常体系的顶层类,其派生出两个重要的子类, Error 和 Exception

2.Erro:是值虚拟机无法解决的问题,比如JVM内部错误,资源耗尽。StackOverflowError和OutOfMemoryError。

  1. Exception:异常产生后程序员可以通过代码进行处理,使程序继续执行。


三、异常的分类

3.1受查异常(编译时异常)

此异常需要对其进行捕获或声明以便于抛出异常。

例如:CloneNotSupportedException(无法克隆异常),IOException(读写异常)

例子:

public class Person Cloneable{
    private String name;
    private String gender;
    int age;
    // 想要让该类支持深拷贝,覆写Object类的clone方法即可
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
}
public class test1 {
    public static void main(String[] args) throws CloneNotSupportedException {
        Person person = new Person();
        Person person1 = (Person) person.clone(); 
        // 此时就会报错,没有声明异常
        // CloneNotSupportedException
    }
}

需要在main方法进行声明异常

public class test1 {
    public static void main(String[] args)  {
        Person person = new Person();
        Person person1 = (Person) person.clone(); 
        // 此时就会报错,没有声明异常
        // CloneNotSupportedException
    }
}

3.2非受查异常(运行时异常)

RunTimeException以及其子类对应的异常,都称为运行时异常。比如:NullPointerException、ArrayIndexOutOfBoundsException、ArithmeticException。

代码:

public class test1 {
    public void main(String[] args[]){
        System.out.println(10/0);  //ArithmeticException。


        int[] array = {1,2,3}
        System.out.println(array[10])//ArrayIndexOutOfBoundsException
    }
}

四、异常的处理

异常处理主要的5个关键字:throw、try、catch、final、throws。


五、异常的抛出

在编写程序时,如果程序中出现错误,此时就需要将错误的信息告知给调用者,比如:参数检测。

在Java中,可以借助throw关键字,抛出一个指定的异常对象,将错误信息告知给调用者。

public void Test(){
    if(// 条件){
        throw new XXXException("异常产生的原因");
    }
}

注意事项:

1. throw必须写在方法体内部

2. 抛出的对象必须是Exception 或者 Exception 的子类对象

3. 如果抛出的是 RunTimeException 或者 RunTimeException 的子类,则可以不用处理,直接交给JVM来处理

4. 如果抛出的是编译时异常,用户必须处理,否则无法通过编译

5. 异常一旦抛出,其后的代码就不会执行

此方法用于自定义异常。

例如:

public void longin(String name, String password){
    if (!this.name.equals(name)){
        throw new UserNameException("大兄弟,用户名输错了");
    }

    if (!this.password.equals(password)){
        throw new PasswordException("大兄弟,密码输错了");
    }
}

六、异常的捕获

异常的捕获,也就是异常的具体处理方式,主要有两种:异常声明throws 以及 try-catch捕获处理。

6.1 throws

处在方法声明时参数列表之后,当方法中抛出编译时异常,用户不想处理该异常,此时就可以借助throws将异常抛

给方法的调用者来处理。即当前方法不处理异常,提醒方法的调用者处理异常。

/*语法格式:
修饰符  返回值类型  方法名(参数列表) throws 异常类型1,异常类型2...{
}*/
public static void main(String[] args) throws CloneNotSupportedException {
}

注意:

1. throws必须跟在方法的参数列表之后

2. 声明的异常必须是 Exception 或者 Exception 的子类

3. 方法内部如果抛出了多个异常,throws之后必须跟多个异常类型,之间用逗号隔开,如果抛出多个异常类型

具有父子关系,直接声明父类即可。

异常处理快捷键:alt + Insert 快速处理

6.2 try-catch

throws对异常并没有真正处理,而是将异常报告给抛出异常方法的调用者,由调用者处理。如果真正要对异常进行

处理,就需要try-catch。

try{
        // 将可能出现异常的代码放在这里
        user.longin("admin1","12345");
    }catch (UserNameException e){
        e.printStackTrace(); // 打印信息最全面
        System.out.println("用户名异常");
    }catch (PasswordException passwordException){
        System.out.println("密码异常");
        passwordException.printStackTrace();
    }finally {
        //必须执行的代码
}

注意:

1. try块内抛出异常位置之后的代码将不会被执行

2. 如果抛出异常类型与catch时异常类型不匹配,即异常不会被成功捕获,也就不会被处理,继续往外抛,直到JVM收到后中断程序----异常是按照类型来捕获的

3. try中可能会抛出多个不同的异常对象,则必须用多个catch来捕获----即多种异常,多次捕获

4.如果异常之间具有父子关系,一定是子类异常在前catch,父类异常在后catch,否则语法错误

5.可以通过一个catch(Exception)捕获所有的异常,即多个异常,一次捕获(不推荐)。

6.当try中存在多个异常的时候,从上往下执行谁先抛出异常就捕获哪个异常,在同一时间,不会出现抛出多个异常,只会排除最try前面的那个异常

6.3 finally

有些特定的代码,不论程序是否发生异常,都需要执行,比如程序中打开的资源:网络连接、数据库连接、IO流等,在程序正常或者异常退出时,必须要对资源进进行回收。另外,因为异常会引发程序的跳转,可能导致有些语句执行不到,finally就是用来解决这个问题的。

注意

finally中的代码一定会执行的,一般在finally中进行一些资源清理的扫尾工作。

6.4面试题

public static void main(String[] args) {
      System.out.println(func());
    }
    public static int func() {
      try {
        return 10;
     } finally {
        return 20;
     }
}

A: 10 B: 20 C: 30 D: 编译失败

输出20

解析:

finally 执行的时机是在方法返回之前(try 或者 catch 中如果有 return 会在这个 return 之前执行 finally). 但是如果

finally 中也存在 return 语句, 那么就会执行 finally 中的 return, 从而不会执行到 try 中原有的 return.

一般我们不建议在 finally 中写 return (被编译器当做一个警告).


七、异常的处理流程

1.程序先执行 try 中的代码

2.如果 try 中的代码出现异常, 就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配.

3.如果找到匹配的异常类型, 就会执行 catch 中的代码

4.如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.

5.无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).

6.如果上层调用者也没有处理的了异常, 就继续向上传递.

7.一直到 main 方法也没有合适的代码处理异常, 就会交给 JVM 来进行处理, 此时程序就会异常终止.


八、自定义异常类

自定义异常通常会继承自 Exception 或者 RuntimeException

继承自 Exception 的异常默认是受查异常

继承自 RuntimeException 的异常默认是非受查异常

work

简单实现登录用户,输入密码,进行捕获异常并抛出。

public class UserNameException extends RuntimeException{
    public UserNameException() {
        super();
    }
    
    // 用于抛出异常的时候显示信息
    public UserNameException(String s) {
        super(s);
    }
}
package work;
import java.util.Scanner;

/**
 * Created with IntelliJ IDEA.
 * Description:异常作业
 * work.User: YAO
 * Date: 2023-02-28
 * Time: 9:57
 */

class User{
    String name = "admin";
    String password = "123456";

    public void Login(String name, String password){
        if (!this.name.equals(name)){
            throw new UserNameException();
        }
        if (!this.password.equals(password)){
            throw new PasswordException();
        }
    }
}
public class work {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入用户名:");
        String  name = scanner.nextLine();
        System.out.println("请输入密码:");
        String password = scanner.nextLine();
        User user = new User();
        try {
            user.Login(name,password);
        }catch (UserNameException e){
            System.out.println("用户名异常:");
            e.printStackTrace();
        }catch (PasswordException passwordException){
            System.out.println("用户密码异常");
            passwordException.printStackTrace();
        }
        finally {
        }
    }
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

哈士奇的奥利奥

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值