java中异常处理机制

简介: java中异常处理机制

一 认识常见异常

在你所写过的代码中,你已经接触过一些异常了,我们可以通过一些简单的代码让我们理解一些简单的异常

1 除0异常(算术异常)

public class winning {
    public static void main(String[] args) {
        System.out.println(10/0);
    }
}

我们都知道0是不能作为除数的,那么在这里作为除数运行之后结果如下:

画出圈的地方就是我们异常的一个种类,称为算术异常

2 数组越界异常

public class winning {
    public static void main(String[] args) {
       int[] array = {1,2,3};
       System.out.println(array[4]);
    }
}

运行结果:

3 空指针异常

public class winning {
    public static void main(String[] args) {
       int[] array = null;
       System.out.println(array[0]);
    }
}

运行结果:

二 异常的体系

Exception是我们所有异常的父类,其中Error以及RuntimeException是我们平时所说的一个非受查异常(运行时异常),其他的异常为受查异常(编译时异常),前面介绍的几种常见的异常,就是RuntimeException的子类。

对于这些异常的处理,我们建议统一利用try-catch去处理,而且catch处理异常时,如果有父子类的异常,我们应该要遵循先写子类的异常,在写父类的异常。


三 异常的用法(try-catch语句)

1概述与代码理解

1 对于编写了一段代码,如果其中的一行发生了异常,那么之后的语句就不再执行(这是因为没有自己处理异常,那么就会交给JVM来处理)

2 对于使用try-catch语句来处理异常,把有可能出现异常的语句放在try中,如果try语句中的一行发生了异常,那么try语句中的下一行将不执行,但不影响后面的语句执行

3 对于try-catch语句,catch后面要跟着异常的类型,以及要相对应的定义一个变量。

public class winning {
    public static void main(String[] args) {
        try {
            int[] array = null;
            System.out.println(array[0]);
            System.out.println("abc");
        }catch(NullPointerException e){
           //打印出异常的调用栈
            e.printStackTrace();
            System.out.println("捕获到空指针异常了");
        }
        System.out.println("执行了");
    }
}

运行结果:

2 注意事项

1 try-catch语句不能单独使用,必须联合起来处理异常

2 可以有多个catch,但不能有多个try

3 catch中可以写多个异常的类型,类型之间需要利用|隔开,但是变量必须定义到所写的最后一个类型的后面(也就是变量只能定义一个),但我们并不推荐这种写法,最好就是一种类型对应一个catch

4 catch中如果没有对应的异常类型,那么此时就会交给jvm去处理( catch 只能处理对应种类的异常)


3 finally语句

finally语句表示的是表示的善后工作,释放资源

public class winning {
    public static void main(String[] args) {
        try {
            int[] array = null;
            System.out.println(array[0]);
            System.out.println("abc");
        }catch(NullPointerException e){
            e.printStackTrace();
            System.out.println("捕获到空指针异常了");
        }finally {
            System.out.println("释放资源");
        }
    }
}

1 其实无论是否存在异常, finally 中的代码一定都会执行到,确保资源可以释放

2 finally语句是按情况写与不写,不是一定要有才能处理异常

3 使用的过程中,要按照try-catch-finally的顺序

利用try负责回收资源(了解)

此时我们可以把鼠标放到try这个关键字上,按下alt + enter,此时就有

83c3b845b5ef4bd9eb994b8282d0de45_watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAamF2Yei_kOiQpQ==,size_20,color_FFFFFF,t_70,g_se,x_16.png

这样子就相当于交由try来负责回收了,但这并不代表每一个异常都可以,只要是try在IDEA中报出了警告,我们可以尝试利用这样的一种快捷键让try去负责回收

finally注意事项

public class winning {
    public static void main(String[] args) {
       System.out.println(func());
    }
    public static int func(){
        try {
            return 20;
        }catch (NullPointerException e){
            e.printStackTrace();
        }finally {
            return 30;
        }
    }
}

运行结果:

30

如果try语句中有返回值,以及finally语句中也有返回值,那么此时将会返回finally语句中的返回值,我们写代码的时候,不要使用这种写法(try与finally中同时拥有返回值)。

4 异常的传递

异常是会随着调用栈的调用而传递

public class winning {
    public static void main(String[] args) {
        try {
            func();
        }catch (ArrayIndexOutOfBoundsException e){
            e.printStackTrace();
        }
    }
    public static void func(){
       int[] array = {1,2,3,4};
       System.out.println(array[29]);
    }
}

运行结果:

如果向上传递的没有合适的方法处理异常,那么就会返回看上一个调用它的有没有合适的方法处理异常,一直到main中也没有合适的处理异常的方法,那么就只能由jvm来处理了

5 throw与throws关键字

对于throw关键字,我们可以通过这个关键字手动抛出异常

public class winning {
    public static void main(String[] args) {
        try {
            func(0);
        }catch (ArrayIndexOutOfBoundsException e){
            e.printStackTrace();
        }
    }
    public static void func(int a){
        if (a==0){
            throw new NullPointerException("空指针异常了");
        }
    }
}

throws 主要是用来提醒调用者,调用该方法可能会出现那些异常,需要进行处理

四 自定义异常

Java 中虽然已经内置了丰富的异常类, 但是我们实际场景中可能还有一些情况需要我们对异常类进行扩展, 创建符合我们实际情况的异常.

我们可以通过一个例子来说明一下:模拟用户登录,此时就可能抛出两种异常,一种是用户名错误异常,一种是密码错误异常:

我们先写出一段模拟登录的代码

public class winning {
    private static final String ID ="java";
    private static final String Password = "666";
    public static void main(String[] args) {
        System.out.println("请输入你的用户名");
        Scanner scanner=new Scanner(System.in);
        String name = scanner.nextLine();
        System.out.println("请输入你的密码");
        String password = scanner.nextLine();
    }
}

定义抛出用户名错误的异常类

//定义一个用户名异常
class NameException extends RuntimeException{
    public NameException(String name){
        super(name);
    }
}

定义抛出密码错误的异常

//定义一个密码异常
class PasswordException extends RuntimeException{
    public PasswordException(String password){
        super(password);
    }
}

我们在进一步完善我们异常处理机制:

public class winning {
    private static final String ID ="java";
    private static final String Password = "666";
    public static void main(String[] args) {
        System.out.println("请输入你的用户名");
        Scanner scanner=new Scanner(System.in);
        String name = scanner.nextLine();
        System.out.println("请输入你的密码");
        String password = scanner.nextLine();
        if (!ID.equals(name)){
            throw new NameException("用户名错了");
        }
        if (!Password.equals(password)){
            throw new PasswordException("密码有误");
        }
    }
}

通过这个例子,让我们了解什么是自定义异常类,如何使用异常类,当然以上代码可以进行优化,可以利用设置一个login登录函数,在利用try-catch进行捕获异常。


注意事项

对于自定义异常类:

1 一定要继承一个异常类(通常为Exception或者RunException)

2 继承Exception说明该类是可受查的

3 继承RunException说明该类是非受查类


结语

关于异常的认识,其实并没有像其他语法一样难理解,如果对你理解异常这一块也有帮助,别忘了给博主点个👍👍哦,之后的学习将会更新一些数据结构的知识!!

目录
相关文章
|
14天前
|
人工智能 Java 关系型数据库
Java——SPI机制详解
SPI(Service Provider Interface)是JDK内置的服务提供发现机制,主要用于框架扩展和组件替换。通过在`META-INF/services/`目录下定义接口实现类文件,Java程序可利用`ServiceLoader`动态加载服务实现。SPI核心思想是解耦,允许不同厂商为同一接口提供多种实现,如`java.sql.Driver`的MySQL与PostgreSQL实现。然而,SPI存在缺陷:需遍历所有实现并实例化,可能造成资源浪费;获取实现类方式不够灵活;多线程使用时存在安全问题。尽管如此,SPI仍是Java生态系统中实现插件化和模块化设计的重要工具。
|
30天前
|
设计模式 人工智能 安全
AQS:Java 中悲观锁的底层实现机制
AQS(AbstractQueuedSynchronizer)是Java并发包中实现同步组件的基础工具,支持锁(如ReentrantLock、ReadWriteLock)和线程同步工具类(如CountDownLatch、Semaphore)等。Doug Lea设计AQS旨在抽象基础同步操作,简化同步组件构建。 使用AQS需实现`tryAcquire(int arg)`和`tryRelease(int arg)`方法以获取和释放资源,共享模式还需实现`tryAcquireShared(int arg)`和`tryReleaseShared(int arg)`。
89 32
AQS:Java 中悲观锁的底层实现机制
|
1月前
|
Java 区块链 网络架构
酷阿鲸森林农场:Java 区块链系统中的 P2P 区块同步与节点自动加入机制
本文介绍了基于 Java 的去中心化区块链电商系统设计与实现,重点探讨了 P2P 网络在酷阿鲸森林农场项目中的应用。通过节点自动发现、区块广播同步及链校验功能,系统实现了无需中心服务器的点对点网络架构。文章详细解析了核心代码逻辑,包括 P2P 服务端监听、客户端广播新区块及节点列表自动获取等环节,并提出了消息签名验证、WebSocket 替代 Socket 等优化方向。该系统不仅适用于农业电商,还可扩展至教育、物流等领域,构建可信数据链条。
|
14天前
|
人工智能 JavaScript Java
Java反射机制及原理
本文介绍了Java反射机制的基本概念、使用方法及其原理。反射在实际项目中比代理更常用,掌握它可以提升编程能力并理解框架设计原理。文章详细讲解了获取Class对象的四种方式:对象.getClass()、类.class、Class.forName()和类加载器.loadClass(),并分析了Class.forName()与ClassLoader的区别。此外,还探讨了通过Class对象进行实例化、获取方法和字段等操作的具体实现。最后从JVM类加载机制角度解析了Class对象的本质及其与类和实例的关系,帮助读者深入理解Java反射的工作原理。
|
1月前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
|
3月前
|
缓存 Dubbo Java
理解的Java中SPI机制
本文深入解析了JDK提供的Java SPI(Service Provider Interface)机制,这是一种基于接口编程、策略模式与配置文件组合实现的动态加载机制,核心在于解耦。文章通过具体示例介绍了SPI的使用方法,包括定义接口、创建配置文件及加载实现类的过程,并分析了其原理与优缺点。SPI适用于框架扩展或替换场景,如JDBC驱动加载、SLF4J日志实现等,但存在加载效率低和线程安全问题。
114 7
理解的Java中SPI机制
|
2月前
|
存储 Java 编译器
Java 中 .length 的使用方法:深入理解 Java 数据结构中的长度获取机制
本文深入解析了 Java 中 `.length` 的使用方法及其在不同数据结构中的应用。对于数组,通过 `.length` 属性获取元素数量;字符串则使用 `.length()` 方法计算字符数;集合类如 `ArrayList` 采用 `.size()` 方法统计元素个数。此外,基本数据类型和包装类不支持长度属性。掌握这些区别,有助于开发者避免常见错误,提升代码质量。
106 1
|
3月前
|
缓存 运维 Java
Java静态代码块深度剖析:机制、特性与最佳实践
在Java中,静态代码块(或称静态初始化块)是指类中定义的一个或多个`static { ... }`结构。其主要功能在于初始化类级别的数据,例如静态变量的初始化或执行仅需运行一次的初始化逻辑。
108 4
|
3月前
|
运维 Java 程序员
Java中的异常处理方法
本文深入剖析Java异常处理机制,介绍可检查异常、运行时异常和错误的区别与处理方式。通过最佳实践方法,如使用合适的异常类型、声明精确异常、try-with-resources语句块、记录异常信息等,帮助开发者提高代码的可靠性、可读性和可维护性。良好的异常处理能保证程序稳定运行,避免资源泄漏和潜在问题。
115 5
|
5月前
|
Java 数据库连接 数据处理
探究Java异常处理【保姆级教程】
Java 异常处理是确保程序稳健运行的关键机制。它通过捕获和处理运行时错误,避免程序崩溃。Java 的异常体系以 `Throwable` 为基础,分为 `Error` 和 `Exception`。前者表示严重错误,后者可细分为受检和非受检异常。常见的异常处理方式包括 `try-catch-finally`、`throws` 和 `throw` 关键字。此外,还可以自定义异常类以满足特定需求。最佳实践包括捕获具体异常、合理使用 `finally` 块和谨慎抛出异常。掌握这些技巧能显著提升程序的健壮性和可靠性。
84 4
OSZAR »