2023.8 - java - Java 异常处理 和 断点

异常是程序中的一些错误,但并不是所有的错误都是异常,并且错误有时候是可以避免的。

比如说,你的代码少了一个分号,那么运行出来结果是提示是错误 java.lang.Error;如果你用System.out.println(11/0),那么你是因为你用0做了除数,会抛出 java.lang.ArithmeticException 的异常。

异常发生的原因有很多,通常包含以下几大类:

  • 用户输入了非法数据。
  • 要打开的文件不存在。
  • 网络通信时连接中断,或者JVM内存溢出。

这些异常有的是因为用户错误引起,有的是程序错误引起的,还有其它一些是因为物理错误引起的。-

要理解Java异常处理是如何工作的,你需要掌握以下三种类型的异常:

  • 检查性异常:最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在文件时,一个异常就发生了,这些异常在编译时不能被简单地忽略。
  • 运行时异常: 运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略。
  • 错误: 错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译也检查不到的。

在Java中,异常分为两大类:检查型异常(Checked Exceptions)和运行时异常(Runtime Exceptions)。

检查型异常(Checked Exceptions)

  • 定义:检查型异常是那些在编译时期就必须被捕获或声明的异常。如果一个方法可能抛出某种类型的检查型异常,那么这个方法必须要么处理这个异常(通过try-catch),要么声明抛出这个异常(通过throws关键字)。
  • 目的:检查型异常的设计初衷是强制程序员处理那些可预见的,可能会在正常程序运行中出现的问题,以增加程序的健壮性。
  • 示例IOExceptionSQLException等。

运行时异常(Runtime Exceptions)

  • 定义:运行时异常是那些在执行时期可能被抛出,但不需要显式捕获或声明的异常。它们继承自RuntimeException类。
  • 目的:运行时异常通常指示编程错误,如试图访问空指针、数组越界等。这类异常通常是可以通过改进程序来避免的。
  • 示例NullPointerExceptionIndexOutOfBoundsExceptionArithmeticException等。

主要区别

  • 处理要求:检查型异常必须在编译时被捕获或通过方法签名声明,而运行时异常则无此要求。
  • 设计哲学:检查型异常用于那些在编写程序时就应该被考虑的情况,以确保程序的健壮性;运行时异常则用于指示程序中的错误,通常是由于程序员的失误导致的。

如何选择

  • 当异常情况是外部错误,且程序员希望调用者必须处理这种情况时,应该使用检查型异常。
  • 当异常情况是由于程序内部错误,如逻辑错误或不正确的使用API导致的,应该使用运行时异常。

子类重写父类方法时,是否可以声明抛出异常(throws),以及可以抛出哪些类型的异常,主要取决于以下几个因素:

  1. 被重写的方法的异常类型:子类重写的方法不能抛出比父类方法更广泛的检查型异常(checked exceptions)。这意味着,如果父类方法没有声明抛出任何检查型异常,那么子类的方法也不能声明抛出检查型异常。如果父类方法声明了抛出某些检查型异常,那么子类方法可以声明抛出相同的异常或这些异常的子类型,也可以不抛出任何异常。

  2. 运行时异常(RuntimeException)和其子类:不受上述规则限制。无论父类方法是否声明抛出异常,子类都可以声明抛出任何运行时异常。

  3. 方法的签名:异常声明是方法签名的一部分,但在重写方法时,子类方法的返回类型、方法名称、参数列表必须与父类方法完全相同。异常类型的规则则遵循上述描述。

  4. 接口默认方法:如果子类实现的接口中有默认方法,并且这个类的父类也有相同的方法,子类在重写这个方法时,对于异常的处理也需要遵循上述规则。

Exception 类的层次

所有的异常类是从 java.lang.Exception 类继承的子类。

Exception 类是 Throwable 类的子类。除了Exception类外,Throwable还有一个子类Error 。

Java 程序通常不捕获错误。错误一般发生在严重故障时,它们在Java程序处理的范畴之外。

Error 用来指示运行时环境发生的错误。

例如,JVM 内存溢出。一般地,程序不会从错误中恢复。

异常类有两个主要的子类:IOException 类和 RuntimeException 类。

 NullPointerException(空指针异常)

是一种常见的运行时异常,通常在Java和其他类似的编程语言中出现。它表示在代码中尝试访问一个空(null)引用的对象时发生了错误。 当你尝试调用一个空引用的方法、访问空引用的属性或者使用空引用执行其他操作时,就会抛出NullPointerException。这通常是由于以下几种情况导致的:

1. 未初始化的引用:如果你尝试使用一个未初始化(赋予null值)的引用,就会抛出空指针异常。
String str;
System.out.println(str.length()); // 会抛出 NullPointerException
2. 调用空引用的方法:如果你尝试调用一个空引用的方法,就会抛出空指针异常。
String str = null;
System.out.println(str.length()); // 会抛出 NullPointerException
3. 对象引用为null:如果你尝试访问一个对象引用的属性或执行其他操作时,而该对象引用为null,就会抛出空指针异常。
String str = null;
System.out.println(str.toUpperCase()); // 会抛出 NullPointerException
为了避免空指针异常,你应该始终确保在使用引用之前,对其进行初始化或者检查其是否为null。可以使用条件语句(如if语句)来检查引用是否为null,或者使用可空类型(如Java 8的Optional)来处理可能为空的引用。
String str = null;
if (str != null) {
    System.out.println(str.length());
} else {
    System.out.println("str is null");
}

OutOfMemoryError(内存溢出错误)
是一种Java运行时错误,表示应用程序在尝试分配内存时无法满足需求,导致内存耗尽。当Java虚拟机(JVM)无法分配更多的内存给应用程序时,就会抛出OutOfMemoryError。

OutOfMemoryError通常由以下几种情况引起:

1. 内存泄漏:当应用程序持续分配内存,但无法释放不再使用的对象时,会导致内存泄漏。随着时间的推移,可用内存逐渐减少,最终导致内存溢出。

2. 大对象或大量对象:如果应用程序需要分配大量的内存来创建大对象或大量的对象,而可用内存不足以容纳它们,就会引发OutOfMemoryError。

3. 递归调用:如果一个递归函数的递归层数过多,每个递归调用都会在堆栈中创建一个新的栈帧,当栈空间耗尽时,就会抛出OutOfMemoryError。

处理OutOfMemoryError的方法取决于具体的情况:

1. 内存泄漏:通过仔细检查代码,确保及时释放不再使用的对象,并使用合适的数据结构和算法来最小化内存占用。

2. 大对象或大量对象:考虑使用对象池或缓存来重复使用对象,避免频繁地创建和销毁对象。

3. 递归调用:优化递归算法,确保递归层数不会过多,或者改为使用迭代循环来替代递归。
package Practices.exception;

/**
 * @author Vince
 * @create 2023-09-13-10:43
 */
public class demo1 {

    int cacu(int a, int b) throws Exception {
        int c = 0;
        System.out.println("计算开始");
        try {
            c = a / b;
        } catch (Exception e) {
            throw e;
        }finally {
            System.out.println("计算结束");
        }
        return c;
    }
}

class   test_demo1{
    public static void main(String[] args) {
        test();
    }

    static void test(){
        try {
            demo1 d = new demo1();
            int a = d.cacu(10,0);
            System.out.println(a);
        } catch (Exception e) {
            System.out.println("异常产生 " + e);
        }
    }
}

 

 如果一个方法 实现一个 【throws Exception】方法,那么该方法可以 捕获异常或者 继续向上抛出异常①,①:此时主方法要 实现【throws Exception】

 Exception 和 RuntimeException 的区别:

①:Exception的异常必须进行捕获,而runtimeException的 异常可以不进行捕获。

(Integer.parseInt(String s)) throws NumberformatException; 

② runtimeException是 Exception的子类 

③:常见的RuntimeException :NullpointException、NumberFormatException、ArithmeticException、ArrayIndexOfBoundsException、ClassCastException。

自定义异常: 

package Practices.exception;

/**
 * @author Vince
 * @create 2023-09-13-13:42
 */
public class myNameException extends Exception {
    myNameException(String msg) {
        super(msg);
    }
}

class  test_myNameException {
    public  static  void  main(String[] args) {
       try {
           throw    new myNameException("hahaha");
       } catch (myNameException e) {
           System.out.println(e);
       }
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值