1. 异常
在java中,阻止当前方法或作用域的情况,称之为异常。
异常发生的原因有很多,通常包含以下几大类:
- 用户输入了非法数据。
- 要打开的文件不存在。
- 网络通信时连接中断,或者JVM内存溢出。
异常有的是因为用户错误引起,有的是程序错误引起的,还有其它一些是因为物理错误引起的。-
2. 异常体系
所有的异常类是从 java.lang.Exception 类继承的子类。
Exception 类是 Throwable 类的子类。除了Exception类外,Throwable还有一个子类Error 。
Java 程序通常不捕获错误。错误一般发生在严重故障时,它们在Java程序处理的范畴之外。
其中Error类中包括虚拟机错误和线程死锁,一旦Error出现了,程序就彻底的挂了,被称为程序终结者;
Error 用来指示运行时环境发生的错误。
例如,JVM 内存溢出。一般地,程序不会从错误中恢复。
异常类有两个主要的子类:IOException 类和 RuntimeException 类。
3. Java 内置异常类
标准运行时异常类的子类是最常见的异常类。由于 java.lang 包是默认加载到所有的 Java 程序的,所以大部分从运行时异常类继承而来的异常都可以直接使用。
非检查性异常
异常 | 描述 |
---|---|
ArithmeticException | 当出现异常的运算条件时,抛出此异常。例如,一个整数"除以零"时,抛出此类的一个实例。 |
ArrayIndexOutOfBoundsException | 用非法索引访问数组时抛出的异常。如果索引为负或大于等于数组大小,则该索引为非法索引。 |
ArrayStoreException | 试图将错误类型的对象存储到一个对象数组时抛出的异常。 |
ClassCastException | 当试图将对象强制转换为不是实例的子类时,抛出该异常。 |
IllegalArgumentException | 抛出的异常表明向方法传递了一个不合法或不正确的参数。 |
IllegalMonitorStateException | 抛出的异常表明某一线程已经试图等待对象的监视器,或者试图通知其他正在等待对象的监视器而本身没有指定监视器的线程。 |
IllegalStateException | 在非法或不适当的时间调用方法时产生的信号。换句话说,即 Java 环境或 Java 应用程序没有处于请求操作所要求的适当状态下。 |
IllegalThreadStateException | 线程没有处于请求操作所要求的适当状态时抛出的异常。 |
IndexOutOfBoundsException | 指示某排序索引(例如对数组、字符串或向量的排序)超出范围时抛出。 |
NegativeArraySizeException | 如果应用程序试图创建大小为负的数组,则抛出该异常。 |
NullPointerException | 当应用程序试图在需要对象的地方使用 null 时,抛出该异常 |
NumberFormatException | 当应用程序试图将字符串转换成一种数值类型,但该字符串不能转换为适当格式时,抛出该异常。 |
SecurityException | 由安全管理器抛出的异常,指示存在安全侵犯。 |
StringIndexOutOfBoundsException | 此异常由 String 方法抛出,指示索引或者为负,或者超出字符串的大小。 |
UnsupportedOperationException | 当不支持请求的操作时,抛出该异常。 |
检查性异常类
异常 | 描述 |
---|---|
ClassNotFoundException | 应用程序试图加载类时,找不到相应的类,抛出该异常。 |
CloneNotSupportedException | 当调用 Object 类中的 clone 方法克隆对象,但该对象的类无法实现 Cloneable 接口时,抛出该异常。 |
IllegalAccessException | 拒绝访问一个类的时候,抛出该异常。 |
InstantiationException | 当试图使用 Class 类中的 newInstance 方法创建一个类的实例,而指定的类对象因为是一个接口或是一个抽象类而无法实例化时,抛出该异常。 |
InterruptedException | 一个线程被另一个线程中断,抛出该异常。 |
NoSuchFieldException | 请求的变量不存在 |
NoSuchMethodException | 请求的方法不存在 |
4. 异常方法
下面的列表是 Throwable 类的主要方法:
序号 | 方法及说明 |
---|---|
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 对象栈层次,添加到栈层次任何先前信息中。 |
5. 捕获异常
使用 try 和 catch 关键字可以捕获异常。try/catch 代码块放在异常可能发生的地方。
try/catch代码块中的代码称为保护代码,使用 try/catch 的语法如下:
try
{
// 程序代码
}catch(ExceptionName e1)
{
//Catch 块
}
Catch 语句包含要捕获异常类型的声明。当保护代码块中发生一个异常时,try 后面的 catch 块就会被检查。
如果发生的异常包含在 catch 块中,异常会被传递到该 catch 块,这和传递一个参数到方法是一样。
package com.zth;
public class Test{
public static void main(String[] args){
try{
int[] a = new int[2];
System.out.println(a[0]);
System.out.println(a[3]);
}catch(ArrayIndexOutOfBoundsException e){
System.out.println("数组下标越界异常" + e);
}
System.out.println("Out of the block" );
}
}
执行结果:
0
数组下标越界异常java.lang.ArrayIndexOutOfBoundsException: 3
Out of the block
6. 多重捕获块
一个 try 代码块后面跟随多个 catch 代码块的情况就叫多重捕获。
多重捕获块的语法如下所示:
try{
// 程序代码
}catch(异常类型1 异常的变量名1){
// 程序代码
}catch(异常类型2 异常的变量名2){
// 程序代码
}catch(异常类型2 异常的变量名2){
// 程序代码
}
上面的代码段包含了 3 个 catch块。
可以在 try 语句后面添加任意数量的 catch 块。
如果保护代码中发生异常,异常被抛给第一个 catch 块。
如果抛出异常的数据类型与 ExceptionType1 匹配,它在这里就会被捕获。
如果不匹配,它会被传递给第二个 catch 块。
如此,直到异常被捕获或者通过所有的 catch 块。
package com.zth;
public class Test{
public static void main(String[] args){
try{
System.out.println(8/0);
String str = null;
str.charAt(0);
}catch(ArithmeticException e){
System.out.println("算术异常");
e.printStackTrace();
}catch(NullPointerException e){
System.out.println("空指针异常" +e);
}catch(Exception e){
System.out.println("异常已被捕获");
}
}
}
执行结果:
算术异常
java.lang.ArithmeticException: / by zero
at com.zth.Test.main(Test.java:6)
7. finally关键字
finally 关键字用来创建在 try 代码块后面执行的代码块。
无论是否发生异常,finally 代码块中的代码总会被执行。
在 finally 代码块中,可以运行清理类型等收尾善后性质的语句。
finally 代码块出现在 catch 代码块最后,语法如下:
try{
// 程序代码
}catch(异常类型1 异常的变量名1){
// 程序代码
}catch(异常类型2 异常的变量名2){
// 程序代码
}finally{
// 程序代码
}
【示例】
package com.zth;
public class Test{
public static void main(String[] args){
try{
System.out.println(8/0);
String str = "zth";
str.charAt(3);
}catch(ArithmeticException e){
System.out.println("算术异常");
e.printStackTrace();
}catch(StringIndexOutOfBoundsException e){
System.out.println("字符串下标越界异常" +e);
}finally{
String str = "zth";
System.out.println(str.charAt(0));
}
}
}
执行结果:
算术异常
java.lang.ArithmeticException: / by zero
z
at com.zth.Test.main(Test.java:6)
【注】
- catch 不能独立于 try 存在。
- 在 try/catch 后面添加 finally 块并非强制性要求的。
- try 代码后不能既没 catch 块也没 finally 块。
- try, catch, finally 块之间不能添加任何代码。
package com.zth;
public class Test{
public static int fun(){
try{
return 1;
}catch(Exception e){
return 2;
}finally{
return 3;
}
}
public static void main(String[] args){
System.out.println(fun()); // 3
}
}
8. throws/throw 关键字
如果一个方法没有捕获到一个检查性异常,那么该方法必须使用 throws 关键字来声明。throws 关键字放在方法签名的尾部。
也可以使用 throw 关键字抛出一个异常,无论它是新实例化的还是刚捕获到的。
每个方法默认抛出 RuntimeException 异常。
一个方法可以声明抛出多个异常,多个异常之间用逗号隔开。
package com.zth;
class Person{
String name;
int age;
public void setName(String name) {
this.name = name;
}
public void setAge(int age) throws RuntimeException {
if(age<0 || age >200){
throw new RuntimeException("年龄不合法");
}else{
this.age = age;
}
}
public int getAge() {
return age;
}
}
public class Test{
public static void main(String[] args){
Person zth = new Person();
zth.setAge(220);
System.out.println(zth.getAge());
}
}
执行结果:
Exception in thread "main" java.lang.RuntimeException: 年龄不合法
at com.zth.Person.setAge(Test.java:12)
at com.zth.Test.main(Test.java:25)
throw与throws的比较
1、throws出现在方法函数头;而throw出现在函数体。
2、throws表示出现异常的一种可能性,并不一定会发生这些异常;throw则是抛出了异常,执行throw则一定抛出了某种异常对象。
3、两者都是消极处理异常的方式(这里的消极并不是说这种方式不好),只是抛出或者可能抛出异常,但是不会由函数去处理异常,真正的处理异常由函数的上层调用处理。
9. 声明自定义异常
- 所有异常都必须是 Throwable 的子类。
- 如果希望写一个检查性异常类,则需要继承 Exception 类。
- 如果你想写一个运行时异常类,那么需要继承 RuntimeException 类。
可以像下面这样定义自己的异常类:
class MyException extends Exception{
}
只继承 Exception 类来创建的异常类是检查性异常类。
package com.zth;
class IllegalAgeException extends RuntimeException{
public IllegalAgeException(){
}
public IllegalAgeException(String str){
super(str);
}
}
class Person{
String name;
int age;
public void setName(String name) {
this.name = name;
}
public void setAge(int age) throws RuntimeException {
if(age<0 || age >200){
throw new IllegalAgeException("年龄不合法");
}else{
this.age = age;
}
}
public int getAge() {
return age;
}
}
public class Test{
public static void main(String[] args){
Person zth = new Person();
zth.setAge(220);
System.out.println(zth.getAge());
}
}
执行结果:
Exception in thread "main" com.zth.IllegalAgeException: 年龄不合法
at com.zth.Person.setAge(Test.java:20)
at com.zth.Test.main(Test.java:33)
10. 综合示例:
老师用电脑讲课:
package com.zth;
import java.util.Random;
class DuanDian extends Error{
public DuanDian(String str){
super(str);
}
}
class CanNotTeachException extends Exception{
public CanNotTeachException(String str){
super(str);
}
}
class Teacher{
public void teach(Computer c) throws CanNotTeachException{
c.start();
try{
c.run();
}catch(Error e){
throw new CanNotTeachException("嗯哼。。断电了");
}
System.out.println("Java...");
c.end();
}
}
class Computer{
public void start(){
System.out.println("开机。。。");
}
public void run() throws Error {
Random ran = new Random();
int i = ran.nextInt(100);
System.out.println("电脑运行中。。。");
if (i > 50) {
throw new DuanDian("断电了。。。");
}
}
public void end(){
System.out.println("电脑关机。。。");
}
}
public class Test{
public static void main(String[] args){
Teacher t = new Teacher();
Computer c = new Computer();
try{
t.teach(c);
}catch (CanNotTeachException e){
e.printStackTrace();
System.out.println("大家下课吧。。。");
}
}
}
执行结果:
开机。。。
电脑运行中。。。
Java...
电脑关机。。。
或
开机。。。
com.zth.CanNotTeachException: 嗯哼。。断电了
电脑运行中。。。
at com.zth.Teacher.teach(Test.java:23)
大家下课吧。。。
at com.zth.Test.main(Test.java:53)