java中的异常 最详细的讲解

一、异常的概念

异常:在程序执行的过程中,出现的非正常情况,最终会导致JVM非正常停止

在Java等面向对象的编程语言中,异常本身是一个类,产生异常就是创建异常对象并抛出了一个异常对象。Java处 理异常的方式是中断处理

异常指的并不是语法错误,语法错了,编译不通过,不会产生字节码文件,根本不能运行.

java中的异常 最详细的讲解_第1张图片Throwable中的常用方法:

public void printStackTrace() :打印异常的详细信息。 包含了异常的类型,异常的原因,还包括异常出现的位置,在开发和调试阶段,都得使用printStackTrace。

public String getMessage() :获取发生异常的原因。 提示给用户的时候,就提示错误原因。

public String toString() :获取异常的类型和异常描述信息(不用)。

二、异常的分类

java中的异常 最详细的讲解_第2张图片

Error:严重错误Error,无法通过处理的错误,只能事先避免

Exception:分为编译器异常和运行时异常

编译期异常,写代码的时候程序出现问题。可以由程序员写的时候解决

Exception也有子类RuntimeException运行期异常,程序运行时可以抛出的异常。

throws抛出异常,交给虚拟机处理,是中断程序执行的

try catch是会把后续代码执行

三、异常的产生过程解析

 java中的异常 最详细的讲解_第3张图片

 四、异常处理

1.抛出异常throw关键字

作用:可以使用ts关键字在指定的方法中抛出指定的异常

格式:throw new xxxException(异常原因)

例如:

throw new NullPointerException("要访问的arr数组不存在");

throw new ArrayIndexOutOfBoundsException("该索引在数组中不存在,已超出范围");

注意:1.throw关键字必须写在方法的内部

2.throw关键字后边new的对象必须是Exception的子类对象

3.throw关键字抛出指定的异常对象,我们就必须处理这个异常对象

throw关键字后边创建的是RuntimeException或RuntimeException的子类对象,我们可以不处理,默认交给JVM处理(打印异常对象,中断程序)

    throw关键字后边创建的是编译异常(写代码报错)就必须处理这个,要么throws要么try..catch

java中的异常 最详细的讲解_第4张图片 NullPointException 空指针异常是运行期异常

 ArrayIndexOutOfBoundsException数组越界异常,运行期异常java中的异常 最详细的讲解_第5张图片

 2.声明异常throws

 异常处理的第一种方式

作用:当方法内部抛出异常对象的时候,那么我们就必须处理这个异常对象

            可以使用throws关键字处理异常对象,会把异常对象声明抛出给方法的调用者处理(自己                不处理,给别人处理)最终交给JVM处理->中断处理

使用格式:修饰符 返回值类型 方法名(参数) throws 异常类名1,异常类名2…{

                 throw new xxxException(异常原因)....

                  }

注意:1.throws必须写在方法声明处

           2.throws后面生命的异常必须是Exception或Exception的子类对象

           3.方法内部如果抛出多个异常对象,那么throws后面也要声明多个异常

              如果抛出的异常有父子类关系,直接声明父类异常就行

           4.调用了一个声明抛出异常的方法,我们就必须处理声明的异常,

                要么继续使用throws声明抛出,交给方法的调用者处理,最终交给JVM

                要么try catch自己处理

public class ThrowsDemo {
public static void main(String[] args) throws FileNotFoundException {
read("a.txt");
}
//定义一个方法,判断文件路径是不是a.txt
// 如果定义功能时有问题发生需要报告给调用者。可以通过在方法上使用throws关键字进行声明
public static void read(String path) throws FileNotFoundException {
if (!path.equals("a.txt")) {//如果不是 a.txt这个文件
// 我假设 如果不是 a.txt 认为 该文件不存在 是一个错误 也就是异常 throw
throw new FileNotFoundException("文件不存在");
//FileNotFoundException是编译时异常
}
}

3.捕获异常try…catch

  异常处理的第二种方式

java中的异常 最详细的讲解_第6张图片

public class TryCatchDemo {
public static void main(String[] args) {
try {// 当产生异常时,必须有处理方式。要么捕获,要么声明。
//try中写可能有异常的语句
read("b.txt");
} catch (FileNotFoundException e) {// 括号中需要定义什么呢?
//try中抛出的是什么异常,在括号中就定义什么异常类型
System.out.println(e);
}
System.out.println("over");
}
/*
*
* 我们 当前的这个方法中 有异常 有编译期异常
*/
public static void read(String path) throws FileNotFoundException {
if (!path.equals("a.txt")) {//如果不是 a.txt这个文件
// 我假设 如果不是 a.txt 认为 该文件不存在 是一个错误 也就是异常 throw
throw new FileNotFoundException("文件不存在");
}
}
}

如何获取异常信息: Throwable类中定义了一些查看方法:

public String getMessage() :获取异常的描述(简短)信息,原因(提示给用户的时候,就提示错误原因。

public String toString() :获取异常的类型和异常描述信息(不用)。

public void printStackTrace() :打印异常的跟踪栈信息并输出到控制台。 JVM打印异常对象,默认此方法,打印的异常信息是最全面的

4.finally 代码块

finally:有一些特定的代码无论异常是否发生,都需要执行。

finally的语法: try...catch....finally:自身需要处理异常,最终还得关闭资源。

注意:1.finally不能单独使用,必须和try一起

           2.finally一般用于资源释放(资源回收),无论程序是否出现异常,最后都要释放资源 

public class TryCatchDemo4 {
public static void main(String[] args) {
try {
read("a.txt");
} catch (FileNotFoundException e) {
//抓取到的是编译期异常 抛出去的是运行期
throw new RuntimeException(e);
} finally {
System.out.println("不管程序怎样,这里都将会被执行。");
}
System.out.println("over");
}
/*
*
* 我们 当前的这个方法中 有异常 有编译期异常
*/
public static void read(String path) throws FileNotFoundException {
if (!path.equals("a.txt")) {//如果不是 a.txt这个文件
// 我假设 如果不是 a.txt 认为 该文件不存在 是一个错误 也就是异常 throw
throw new FileNotFoundException("文件不存在");
}
}
}

五、异常注意事项

多个异常使用捕获又该如何处理呢?

1. 多个异常分别处理。

2. 多个异常一次捕获,多次处理。

3. 多个异常一次捕获一次处理。

般我们是使用一次捕获多次处理方式,格式如下:

try{
编写可能会出现异常的代码
}catch(异常类型A e){ 当try中出现A类型异常,就用该catch来捕获.
处理异常的代码
//记录日志/打印异常信息/继续抛出异常
}catch(异常类型B e){ 当try中出现B类型异常,就用该catch来捕获.
处理异常的代码
//记录日志/打印异常信息/继续抛出异常
}

注意:这种异常处理方式,要求多个catch中的异常不能相同,并且若catch中的多个异常之间有子父类异 常的关系,那么子类异常要求在上面的catch处理,父类异常在下面的catch处理

原因如下:java中的异常 最详细的讲解_第7张图片

运行时异常被抛出可以不处理。即不捕获也不声明抛出。交给JVM处理

如果finally有return语句,永远返回finally中的结果,避免该情况.

java中的异常 最详细的讲解_第8张图片

如果父类抛出了多个异常,子类重写父类方法时,抛出和父类相同的异常或者是父类异常的子类或者不抛出异常。  父类方法没有抛出异常,子类重写父类该方法时也不可抛出异常。此时子类产生该异常,只能捕获处理,不 能声明抛出。(也就是说,父类异常是什么样,子类异常就是什么样)

java中的异常 最详细的讲解_第9张图片

java中的异常 最详细的讲解_第10张图片

 六、自定义异常

为什么需要自定义异常类:java提供的异常类,不够我们使用,需要自己定义一些

格式:public class xxxException extends Exception/RuntimeException  {

              添加一个空参数构造方法

               添加一个带异常信息的构造方法}

异常类如何定义: 1. 自定义一个编译期异常: 自定义类 并继承于 java.lang.Exception 。 2. 自定义一个运行时期的异常类:自定义类 并继承于 java.lang.RuntimeException 。

// 业务逻辑异常
public class RegisterException extends Exception {
/**
* 空参构造
添加一个空参数构造方法
*/
public RegisterException() {
}
/**
*
* @param message 表示异常提示
添加一个带异常信息的构造方法
*/
//查看源码发现,所有的异常类都会有一个带异常信息的构造方法,方法内部会调用父类带异常信息的构造方法,让父类来处理这个异常信息
public RegisterException(String message) {
super(message);
}
}

你可能感兴趣的:(java,java)