异常:在程序执行的过程中,出现的非正常情况,最终会导致JVM非正常停止
在Java等面向对象的编程语言中,异常本身是一个类,产生异常就是创建异常对象并抛出了一个异常对象。Java处 理异常的方式是中断处理
异常指的并不是语法错误,语法错了,编译不通过,不会产生字节码文件,根本不能运行.
public void printStackTrace() :打印异常的详细信息。 包含了异常的类型,异常的原因,还包括异常出现的位置,在开发和调试阶段,都得使用printStackTrace。
public String getMessage() :获取发生异常的原因。 提示给用户的时候,就提示错误原因。
public String toString() :获取异常的类型和异常描述信息(不用)。
Error:严重错误Error,无法通过处理的错误,只能事先避免
Exception:分为编译器异常和运行时异常
编译期异常,写代码的时候程序出现问题。可以由程序员写的时候解决
Exception也有子类RuntimeException运行期异常,程序运行时可以抛出的异常。
throws抛出异常,交给虚拟机处理,是中断程序执行的
try catch是会把后续代码执行
作用:可以使用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
NullPointException 空指针异常是运行期异常
ArrayIndexOutOfBoundsException数组越界异常,运行期异常
异常处理的第一种方式
作用:当方法内部抛出异常对象的时候,那么我们就必须处理这个异常对象
可以使用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是编译时异常 } }
异常处理的第二种方式
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打印异常对象,默认此方法,打印的异常信息是最全面的
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处理。
运行时异常被抛出可以不处理。即不捕获也不声明抛出。交给JVM处理
如果finally有return语句,永远返回finally中的结果,避免该情况.
如果父类抛出了多个异常,子类重写父类方法时,抛出和父类相同的异常或者是父类异常的子类或者不抛出异常。 父类方法没有抛出异常,子类重写父类该方法时也不可抛出异常。此时子类产生该异常,只能捕获处理,不 能声明抛出。(也就是说,父类异常是什么样,子类异常就是什么样)
为什么需要自定义异常类: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);
}
}