Java之异常的详细总结

今天来总结一下Java的异常相关知识点


1.异常

定义: 在Java中,将程序执行过程中发生的一些不正常行为称之为"异常"。常见的两大异常有越界异常(IndexOutOfBoundsException)和空指针异常(NullPointerException)。


2.异常的继承体系

继承体系图:
在这里插入图片描述

从上图中可以看到:

  1. Throwable:是异常体系的顶层类,其派生出两个重要的子类, Error 和 Exception
  2. Error:指的是Java虚拟机无法解决的严重问题,比如:JVM的内部错误、资源耗尽等,典型代表:StackOverflowError和OutOfMemoryError,一旦发生回力乏术,程序就会直接退出。
  3. Exception:异常产生后程序员可以通过代码进行处理,使程序继续执行。比如:感冒、发烧。我们平时所说的异常就是Exception。

异常的分类(根据异常的发生时机划分):

受查异常(编译时异常)︰在程序编译阶段需要显示进行异常处理的异常类,左图中红色框都属于受查异常,常见的有IOException(IO相关的异常),SQLException(数据库操作相关的异常)。 受查异常必须显示处理,处理方式有两种:1.使用try…catch处理;2.使用throws抛出给JVM处理。这两种方法下面会介绍。

非受查异常(运行时异常)︰在程序运行时发生的异常,不需要显示进行异常处理的异常都是非受查异常。左图中蓝色框都是非受查异常。Error与RuntimeException都是非受查异常,NPE,越界,算术运算都是非受查异常。


2.异常的处理流程

异常处理格式:

try{
     //所有可能产生异常的代码块
     //当异常产生时,当前语句后面的语句不再执行,去catch中匹配具体的异常类
}catch(要捕获的异常类型){//若没有相关的catch块与异常匹配,则程序退出
     //出现异常之后执行的代码
}[final{
     //无论是否发生异常,一定会执行的代码块
     //一般资源关闭操作,网络的关闭操作都放在finally代码块中,保证资源一定会被关闭!!!
}]   //final为可选代码块(用[]表示了可选可不选) 可写可不写

try…catch代码演示:

1:ArithmeticException(算术异常)

    public static void main(String[] args) {
        System.out.println("异常产生前的代码");
        try {
            System.out.println(10 / 0);
        } catch (ArithmeticException e) {
            System.out.println("异常产生了!!!");
        }
        System.out.println("异常产生后的代码");
    }

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


2:若catch中的类型与异常类型不匹配

    public static void main(String[] args) {
        System.out.println("异常产生前的代码");
        try {
//            System.out.println(10 / 0);
            String str = null;
            System.out.println(str.length());
        } catch (ArithmeticException e) {
            System.out.println("异常产生了!!!");
        }
        System.out.println("异常产生后的代码");
    }

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


3:可能产生多个异常的代码的处理方式

方式一:多个catch捕获

    public static void main(String[] args) {
        System.out.println("异常产生前的代码");
        try {
            System.out.println(10 / 0);
            String str = null;
            System.out.println(str.length());
        } catch (ArithmeticException e) {
            System.out.println("算术异常产生了!!!");
        }catch (NullPointerException e){
            System.out.println("空指针异常产生了!!!");
        }
        System.out.println("异常产生后的代码");
    }

结果: 正好验证了上面说的//当异常产生时,当前语句后面的语句不再执行,去catch中匹配具体的异常类这句话,只输出了算术异常,没有输出空指针异常。
在这里插入图片描述

方式二:使用多态方式,将catch中的异常类改为Exception e,捕获所有子类异常。

    public static void main(String[] args) {
        System.out.println("异常产生前的代码");
        try {
            System.out.println(10 / 0);
            String str = null;
            System.out.println(str.length());
        } catch (Exception e) {
            System.out.println("异常产生了!!!");
        }
        System.out.println("异常产生后的代码");
    }

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

二者选择:推荐使用多个catch来捕获异常,因为多个catch捕获异常我们可以知道该程序段发生了何种异常,便于维护。但是我们可以在循环捕获的最后一个catch里面写Exception来做为保底。


throws和throw抛出异常:

throws:出现在方法头部,方法的定义上使用 throws 表示这个方法可能抛出某种异常,若throws抛出多个异常,若抛出的异常有父子关系,只需要声明父类异常即可,该抛出需要由方法的调用者进行异常处理。

throw:出现在方法内部,表示方法内抛出某种异常对象,如果异常对象是非 RuntimeException 则需要在方法申明时加上该异常的抛出 即需要加上 throws 语句 或者 在方法体内 try catch 处理该异常,否则编译报错执行到 throw 语句则后面的语句块不再执行。


补充介绍printStackTrace方法:

我们如何像JVM那样把产生的异常位置返回输出告诉我们呢?这里我们需要使用到printStackTrace方法,该方法捕获到相关异常之后打印异常产生的错误原因以及出现的位置。

    public static void main(String[] args) {
        System.out.println("异常产生前的代码");
        try {
            System.out.println(10 / 0);
        } catch (ArithmeticException e) {
            System.out.println("异常产生了!!!");
            e.printStackTrace();
        }
        System.out.println("异常产生后的代码");
    }

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

这里大家或许会问:这打印顺序不对呀,printStackTrace方法输出不应该是在 “异常产生了!!!” 之后吗?

解释:printStackTrace方法调用的输出是System.err.println(),这是一种错误输出,它的输出在控制台为红色字体,且它的优先级低于标准输出System.out.println()

验证:

    public static void main(String[] args) {
        System.out.println("异常产生前的代码");
        try {
            System.out.println(10 / 0);
        } catch (ArithmeticException e) {
            System.err.println("异常产生了!!!");
            e.printStackTrace();
        }
        System.out.println("异常产生后的代码");
    }

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

我将 “异常产生了!!!” 设置成错误输出,这时它就和 printStackTrace()方法的输出 一起在末尾输出了。


4.自定义异常类型

JDK中内置了很多异常类,但是在程序开发过程中,很多场景下出现的问题,JDK中没有相关的异常对应,此时我们就需要根据自己的需求来自定义异常类。

自定义异常类型两步走:
1.只需要继承Exception(受查)或者RuntimeException(非受查)——>必须显示处理异常的继承Exception,可以不显示处理,就继承RuntimeException。
2.实现一个带String的有参构造,String方法参数,就是异常的原因。

用户登录案例:

public class UserLogin {
    private String userName = "admin";
    private String password = "123456";

    public static void main(String[] args) {
        UserLogin userLogin = new UserLogin();
        try {
            userLogin.login("admin", "1234567");
        } catch (PasswordException e) {
            e.printStackTrace();
        }
    }

    public void login(String userName, String password) throws PasswordException {
        if (!this.userName.equals(userName)) {
            throw new UserNameException("用户名错误!");
        }
        if (!this.password.equals(password)) {
            // 产生异常对象之后需要显示处理
            // 显示处理有两种,要么在当前方法try...catch
            // 要么通过throws向上抛出
            throw new PasswordException("密码错误!");
        }
    }

}

// 密码异常,受查异常,在main里面显示处理了
class PasswordException extends Exception {
    public PasswordException(String msg) {
        super(msg);
    }
}

// 用户名异常,非受查异常
class UserNameException extends RuntimeException {
    public UserNameException(String msg) {
        super(msg);
    }
}

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
其他还有很多异常,我就不一一列举了,我要说明的是,一个合格的程序员,需要对程序中常见的问题有相当的了解和相应的解决办法,否则仅仅停留在写程序而不会改程序的话,会极大影响到自己的开发的。关于异常的全部说明,在api里都可以查阅。 算术异常类:ArithmeticExecption 空指针异常类:NullPointerException 类型强制转换异常:ClassCastException 数组负下标异常:NegativeArrayException 数组下标越界异常:ArrayIndexOutOfBoundsException 违背安全原则异常:SecturityException 文件已结束异常:EOFException 文件未找到异常:FileNotFoundException 字符串转换为数字异常:NumberFormatException 操作数据库异常:SQLException 输入输出异常:IOException 方法未找到异常:NoSuchMethodException java.lang.AbstractMethodError 抽象方法错误。当应用试图调用抽象方法时抛出。 java.lang.AssertionError 断言错。用来指示一个断言失败的情况。 java.lang.ClassCircularityError 类循环依赖错误。在初始化一个类时,若检测到类之间循环依赖则抛出该异常java.lang.ClassFormatError 类格式错误。当Java虚拟机试图从一个文件中读取Java类,而检测到该文件的内容不符合类的有效格式时抛出。 java.lang.Error 错误。是所有错误的基类,用于标识严重的程序运行问题。这些问题通常描述一些不应被应用程序捕获的反常情况。 java.lang.ExceptionInInitializerError 初始化程序错误。当执行一个类的静态初始化程序的过程中,发生了异常时抛出。静态初始化程序是指直接包含于类中的static语句段。 java.lang.IllegalAccessError 违法访问错误。当一个应用试图访问、修改某个类的域(Field)或者调用其方法,但是又违反域或方法的可见性声明,则抛出该异常java.lang.IncompatibleClassChangeError 不兼容的类变化错误。当正在执行的方法所依赖的类定义发生了不兼容的改变时,抛出该异常。一般在修改了应用中的某些类的声明定义而没有对整个应用重新编译而直接运行的情况下,容易引发该错误。 java.lang.InstantiationError 实例化错误。当一个应用试图通过Java的new操作符构造一个抽象类或者接口时抛出该异常. java.lang.InternalError 内部错误。用于指示Java虚拟机发生了内部错误。 java.lang.LinkageError 链接错误。该错误及其所有子类指示某个类依赖于另外一些类,在该类编译之后,被依赖的类改变了其类定义而没有重新编译所有的类,进而引发错误的情况。 java.lang.NoClassDefFoundError 未找到类定义错误。当Java虚拟机或者类装载器试图实例化某个类,而找不到该类的定义时抛出该错误。 java.lang.NoSuchFieldError 域不存在错误。当应用试图访问或者修改某类的某个域,而该类的定义中没有该域的定义时抛出该错误。 java.lang.NoSuchMethodError 方法不存在错误。当应用试图调用某类的某个方法,而该类的定义中没有该方法的定义时抛出该错误。 java.lang.OutOfMemoryError 内存不足错误。当可用内存不足以让Java虚拟机分配给一个对象时抛出该错误。 java.lang.StackOverflowError 堆栈溢出错误。当一个应用递归调用的层次太深而导致堆栈溢出时抛出该错误。 java.lang.ThreadDeath 线程结束。当调用Thread类的stop方法时抛出该错误,用于指示线程结束。 java.lang.UnknownError 未知错误。用于指示Java虚拟机发生了未知严重错误的情况。 java.lang.UnsatisfiedLinkError 未满足的链接错误。当Java虚拟机未找到某个类的声明为native方法的本机语言定义时抛出。 java.lang.UnsupportedClassVersionError 不支持的类版本错误。当Java虚拟机试图从读取某个类文件,但是发现该文件的主、次版本号不被当前Java

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

木木是木木

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

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

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

打赏作者

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

抵扣说明:

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

余额充值