Java 基础-异常处理
在 Java 中声明了很多异常类,每个异常类都表示一种运行错误。程序运行过程中发生一个可识别的运行错误时(可以找到与错误匹配的异常类,例如被除数为 0 时会触发 java.lang.ArithmeticException
),系统会抛出对应异常类的对象。
参考:Java 异常处理
Java 异常处理机制的优点
- 分离错误处理代码,使业务代码更专注
- 按照类型对错误分组
- 可以捕获处理无法预测的错误
- 异常类的对象包含了异常的充分信息
- 可以按照调用栈传播错误,直到有处理错误的代码
错误分类
根据错误的严重程度不同,可以分为两类:
- 错误:致命的,无法处理的。最上层父类是 Error 类
- 异常:非致命,可以捕获并处理。最上层父类是 Exception 类
Throwable 是 Error 和 Exception 的父类。
Java 中有3种方式生成异常对象:
- 由 Java 虚拟机生成
- 由 Java 类库中的某些类生成
- 在自己写的程序中生成和抛出异常对象
必须通过 throw 语句抛异常对象,异常对象必须是 Throwable 或其子类的实例。
throw new XXException();
XXException e = new XXException();
throw e;
Throwable 类的主要方法
方法 | 描述 |
---|---|
public String getMessage() | 获取异常的详细信息 |
public Throwable getCause() | 返回 Throwable 对象,代表异常的原因 |
public void printStackTrace() | 打印toString()结果和栈层次到 System.err 错误输出流。 |
public StackTraceElement [] getStackTrace() | 返回一个包含堆栈层次的数组。下标为0的元素代表栈顶,最后一个元素代表方法调用堆栈的栈底。 |
异常的分类
根据是否必须捕获异常,可以将异常分为两类:
非检查型异常(也叫运行时异常)
非检查型异常继承自 RuntimeException,不需要在程序中进行捕获,编译器也不会检查。之所以可以不捕获,有两个原因:
- 引发这类异常的操作经常出现(例如使用对象时如果对象为 null 则抛异常)
- 要抛的异常可以用其他方式解决(例如被除数为0会抛出异常,但可以提前判断被除数来防止这种异常)
public class Excep {
public static void main(String[] args) {
String[] arr = {"hello", "world"};
int i = 0;
while(i < 10) {
System.out.println(arr[i++]);
}
}
}
上面这段代码并没有捕获异常,编译通过,执行时会因为数组下标越界而报错:
[root@VM_139_38_centos java]# java Excep
hello
world
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 2
at Excep.main(Excep.java:6)
检查型异常
对于检查型异常,调用者必须捕获并处理(try{...}catch(...){...}
)或也声明抛出(`throws XXException``)。如果所有方法都没有处理这种异常,最终异常会交给 Java 虚拟机来处理。编译器会检查这种异常。
public class T {
public static void main(String[] args) {
He h = new He();
h.fun();
}
}
class He {
public void fun() throws Exception {
throw new RuntimeException();
}
}
上面这段代码中,类 He 中的 fun 方法会抛出异常,当时调用者并没有捕获也没有继续抛出,所以执行时会报错:
T.java:4: unreported exception java.lang.Exception; must be caught or declared to be thrown
h.fun();
^
1 error
如果异常一路都在抛,没有方法处理,最后到了 main 方法还在抛异常public static void main(String[] args) throws Exception{...}
,那最后 Java 虚拟机会接收到这个异常,程序会停止执行并报错,例如:
public class T {
public static void main(String[] args) throws Exception{
He h = new He();
h.fun();
System.out.println("end of program");
}
}
class He {
public void fun() throws Exception {
throw new RuntimeException();
}
}
[root@VM_139_38_centos java]# java T
Exception in thread "main" java.lang.RuntimeException
at He.fun(T.java:10)
at T.main(T.java:4)
捕获并处理异常
Java 中用 try{...}catch(...){...}finally{...}
语句捕获异常。其中,try 代码段中是可能抛出异常的代码,catch 代码段在匹配异常时执行,finally 代码段则是无论是否发生异常都会执行的代码段。语法为:
try {
// 程序代码
} catch(ExceptionName1 e1) {
//Catch 块
} catch(ExceptionName2 e2) {
//可以有多个 Catch 块
} finally {
//finally 块
}
例如,对于数组越界异常的捕获处理:
public class ExcepTest{
public static void main(String args[]){
try{
int a[] = new int[2];
System.out.println("Access element three :" + a[3]);
}catch(ArrayIndexOutOfBoundsException e){
System.out.println("Exception thrown :" + e);
}finally{
System.out.println("finally run");
}
System.out.println("Out of the block");
}
}
结果为:
Exception thrown :java.lang.ArrayIndexOutOfBoundsException: 3
finally run
Out of the block
throws/throw 关键字
对于检查性异常,如果方法不想捕获,那么该方法必须使用 throws 关键字来抛出这个异常(如果有多个可能的异常,需要用逗号分隔),交给调用者来处理。throws 关键字放在方法签名的尾部。
public void fun() throws RemoteException, InsufficientFundsException
{
// Method implementation
}
throw 关键字用于抛出异常给方法的调用者,注意需要在方法签名尾部用 throws
抛出这个异常。
public void fun() throws RemoteException
{
throw new RemoteException();
}
自定义异常
Java 中的异常类跟普通的类一样,有属性和方法。根据自定义的异常是否需要检查,可以分为两类:
- 检查型异常类,需要继承 Exception 类。
- 运行时异常类,需要继承 RuntimeException 类。
语法示例:
class MyException extends Exception{
}
代码示例:
public class T {
public static void main(String[] args) {
try {
throw new MyException(666);
} catch (MyException e) {
System.out.println(e);
}
}
}
class MyException extends Exception {
private int amount;
public MyException(int amount) {
this.amount = amount;
}
public String getMessage() {
return "amount not enough" + this.amount;
}
}
运行结果:
MyException: amount not enough666
最新文章
- VoxelGrid体素滤波器对点云进行下采样
- java常用linux命令
- POJ_3104_Drying_(二分,最小化最大值)
- Intra-cluster Replication in Apache Kafka--reference
- (转)fastcgi协议的简单实现
- ThinkPHP使用Memcached缓存数据
- Python自学笔记——Matplotlib风羽自定义
- 【机器学习】代价函数(cost function)
- tkinter模块常用参数(python3)
- pycharm和shell中的sys.path不一样
- 表单/iframe与video标签
- Centos7的防火墙关闭
- 软件测试人员需要掌握的linux命令(二)
- #2 Python面向对象(一)
- sql语句-单表查询
- Paramiko和堡垒机实现
- splash 安装
- java学习笔记27(File类)
- Java——如何创建文件夹及文件,删除文件,文件夹
- MySQL是如何利用索引的