java异常处理
概念
异常是导致程序中断运行的一种指令流,如果不对异常进行正确的处理,则可能导致程序中断,造成不必要的损失所以在程序的设计中必须要考虑各种异常的发生,并正确的做好相应的处理,这样才能保证程序正常的执行。
异常是程序运行时产生的一些错误,错误不一定全是异常,有些异常则可以避免。
例如一些代码的语法错误是可以避免的
主要的异常有:
- 检查性异常:最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在文件时,一个异常就发生了,这些异常在编译时不能被简单地忽略。
- 运行时异常: 运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略。
- 错误: 错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译也检查不到的。
在整个java的长结构整,实际上有两种最常用的类分别为Exception和Error,这两个都是Throwabke的子类,
如下图所示
Exception: 一般表示的是程序中出现的问题,可以通过try…catch处理
Error:主要是用来指示运行环境发生的错误,程序中无法处理
一般情况下,开发者习惯将Exception和Error统称为异常
下面的表中列出了 Java 的非检查性异常。
Java 内置异常类
异常 | 描述 |
---|---|
ArithmeticException | 当出现异常的运算条件时,抛出此异常。例如,一个整数”除以零”时,抛出此类的一个实例。 |
ArrayIndexOutOfBoundsException | 用非法索引访问数组时抛出的异常。如果索引为负或大于等于数组大小,则该索引为非法索引。 |
ArrayStoreException | 试图将错误类型的对象存储到一个对象数组时抛出的异常。 |
ClassCastException | 当试图将对象强制转换为不是实例的子类时,抛出该异常。 |
IllegalArgumentException | 抛出的异常表明向方法传递了一个不合法或不正确的参数。 |
IllegalMonitorStateException | 抛出的异常表明某一线程已经试图等待对象的监视器,或者试图通知其他正在等待对象的监视器而本身没有指定监视器的线程。 |
IllegalStateException | 在非法或不适当的时间调用方法时产生的信号。换句话说,即 Java 环境或 Java 应用程序没有处于请求操作所要求的适当状态下。 |
IllegalThreadStateException | 线程没有处于请求操作所要求的适当状态时抛出的异常。 |
IndexOutOfBoundsException | 指示某排序索引(例如对数组、字符串或向量的排序)超出范围时抛出。 |
NegativeArraySizeException | 如果应用程序试图创建大小为负的数组,则抛出该异常。 |
NullPointerException | 当应用程序试图在需要对象的地方使用 null 时,抛出该异常 |
NumberFormatException | 当应用程序试图将字符串转换成一种数值类型,但该字符串不能转换为适当格式时,抛出该异常。 |
SecurityException | 由安全管理器抛出的异常,指示存在安全侵犯。 |
StringIndexOutOfBoundsException | 此异常由 String 方法抛出,指示索引或者为负,或者超出字符串的大小。 |
UnsupportedOperationException | 当不支持请求的操作时,抛出该异常。 |
- 下面的表中列出了 Java 定义在 java.lang 包中的检查性异常类。
异常 | 描述 |
---|---|
ClassNotFoundException | 应用程序试图加载类时,找不到相应的类,抛出该异常。 |
CloneNotSupportedException | 当调用 Object 类中的 clone 方法克隆对象,但该对象的类无法实现 Cloneable 接口时,抛出该异常。 |
IllegalAccessException | 拒绝访问一个类的时候,抛出该异常。 |
InstantiationException | 当试图使用 Class 类中的 newInstance 方法创建一个类的实例,而指定的类对象因为是一个接口或是一个抽象类而无法实例化时,抛出该异常。 |
InterruptedException | 一个线程被另一个线程中断,抛出该异常。 |
NoSuchFieldException | 请求的变量不存在 |
NoSuchMethodException | 请求的方法不存在 |
异常方法
序号 | 方法及说明 |
---|---|
1 | public String getMessage()返回关于发生的异常的详细信息。这个消息在Throwable 类的构造函数中初始化了。 |
2 | public Throwable getCause()返回一个Throwable 对象代表异常原因。 |
3 | public String toString()使用getMessage()的结果返回类的串级名字。 |
4 | public void printStackTrace()打印toString()结果和栈层次到System.err,即错误输出流。 |
5 | public StackTraceElement [] getStackTrace()返回一个包含堆栈层次的数组。下标为0的元素代表栈顶,最后一个元素代表方法调用堆栈的栈底。 |
6 | public Throwable fillInStackTrace()用当前的调用栈层次填充Throwable 对象栈层次,添加到栈层次任何先前信息中。 |
捕获异常
使用 try 和 catch 关键字可以捕获异常。try/catch 代码块放在异常可能发生的地方。
try {
//你自己的代码
} catch (异常类型 异常变量) {
//catch
}
如果一个异常满足不了,你还可以进行多重捕获异常
try {
//你自己的代码
} catch (异常类型 异常变量) {
//catch
}catch (异常类型 异常变量) {
//catch
}catch (异常类型 异常变量) {
//catch
}catch (异常类型 异常变量) {
//catch
}
throws关键字:
在定义一个方法是可以使用throws关键字声明,使用throws声明的方法表示此方法不处理异常,而交方法的调用处理进行处理
例:
class Math{
public int div(int a,int b) throws Exception{
int temp=a/b;
return remp;
}
}
以为除法操作有可能出现异常,也有可能没有异常,所以在上面的方法使用了throws关键字,表示不管是否会有异常,在调用此方法都必须进行异常处理
class Math{
public int div(int a,int b) throws Exception{//该方法可以不处理异常
int temp=a/b;//有可能出现异常
return remp;//返回计算结果
}
}
public class Test{
public static void main(String args []){
Math m=new Math();//实例化Math对象
try{//因为有throws。不管是否有异常,都必须处理
System.out.println("除法:"+m.div(10,2));
}catch(Exception e){
e.printStackTrace();//打印异常信息
}
}
}
运行结果:
除法:5
throws是在方法处定义的,那么在主方法也可以使用throws关键字,但是主方法是程序的起点,所以此时主方法再向上抛出异常,则只能交给JVM进行处理。就好比我们部门处理不了的问题,要上报给经理,经理处理不了上报给董事长,到了董事长就到头了,那么java的“头”就是JVM
class Math{
public int div(int a,int b) throws Exception{//该方法可以不处理异常
int temp=a/b;//有可能出现异常
return remp;//返回计算结果
}
}
public class Test{
//本方法中的异常都可以不适用try...catch处理
public static void main(String args []) throws Exception{
Math m=new Math();//实例化Math对象
System.out.println("除法:"+m.div(10,2));
}
}
运行结果:
除法:5
throw关键字
与throws不同的是,可以直接使用throw抛出一个一餐个,抛出时直接抛出异常类的实例化对象即可。
public class Test{
public static void main(String args []){
try{
throw new Exception("自己抛出的异常!");//抛出异常的实例化对象
}catch(Exception e){//捕获异常
System.out.println(e);
}
}
}
运行结果:
java.lang.Exception:自己抛出的异常
注意:throw不会单独使用
finally关键字
finally 关键字用来创建在 try 代码块后面执行的代码块。
无论是否发生异常,finally 代码块中的代码总会被执行。
在 finally 代码块中,可以运行清理类型等收尾善后性质的语句。
try {
//你自己的代码
} catch (异常类型 异常变量) {
//catch
}finally{
// 代码
}
- 注意下面事项:
- catch 不能独立于 try 存在。
- 在 try/catch 后面添加 finally 块并非强制性要求的。
- try 代码后不能既没 catch 块也没 finally 块。
- try, catch, finally 块之间不能添加任何代码。
自定义异常类
在java中已经提供了大量的异常类,但是这些异常类有时也很难满足自己的开发需求,那么这是就需要开发者自定义异常类来满足自己的需求了。自定义异常类只需要继承Exception类即可。
例:
class MyException extends Exception{/异常类,继承Exception类
public MyException (String msg){//构造方法接收异常信息
super(msg);//调用父类中的构造方法
}
}
public calss Test{
public static void main(String args []){
try{
throw new MyException("自定义异常");//抛出异常
}catch (Exception e){//异常处理
System.out.println(e);
}
}
}
运行结果:
MyException:自定义异常
总结
1. 异常是导致程序中断运行的一种指令流,当异常发生时,如果没有进行良好的处理则程序会中断执行。
2. 异常可以使用try…catch进行处理,也可以使用try…catch…finally进行处理,在try语句中捕捉异常,然后在catch中处理异常,finally作为异常的同意出口,不管是否发生异常都要执行此代码段。
3. 异常的最大父类是Throwable,其子类为Exception和Error。Exception表示程序处理的异常,而Error表示JVM错误,一般不有程序开发人员处理
4. 发生异常后,JVM会自动产生一个异常类的实例化对象,并匹配相应的catch语句中的异常类型,也可以利用对象的向上转型关系直接捕获Exception。
5. throws用在方法声明处,表示本方法不处理异常
6. throw表示方法中手工抛出一个异常