异常分两种大的异常类型,运行时异常和受检查异常。
用户既可以使用系统的异常类来处理异常信息,也可以创建系统的异常类的子类来自定义异常,这种方式比较灵活,虚拟机可以报出自己设置的异常信息,清楚明白。
1、运行时异常
运行时异常的特点时java编译器不去检查它,也就是说,当程序中可能出现这类异常时,即时没有有try---catch语句捕获它,也没有使用throws语句抛出它,还是会编译通过。
2、受检查异常
除了运行异常外,其他异常都属于受检查异常,这种异常的特点是要么用try...catch捕获处理,要么用throws语句声明抛出,否则编译不会通过。
3、两者的区别
运行时异常表示无法让程序恢复运行的异常,导致这种异常的原因通常是由于执行了错误的操作。一旦出现错误,建议让程序终止。
受检查异常表示程序可以处理的异常。如果抛出异常的方法本身不处理或者不能处理它,那么方法的调用者就必须去处理该异常,否则调用会出错,连编译也无法通过。当然,这两种异常都是可以通过程序来捕获并处理的。
对于运行异常,建议不要用try...catch...捕获处理,应该在程序开发调试的过程中尽量的避免,当然有一些必须要处理的,自己知道了那个部分会出现异常,而这种异常你要把它处理的你想要的结果,例如:空值处理。
异常的处理方式有三种:(1)用户完全自己处理、(2)抛给主方法,主方法再抛给虚拟机处理、(3)先抛出来,用户再自己处理
使用系统异常举例如下:
第一种处理方式(用户自己检测、捕获、处理):
class ExceptionDemo { public int method(int a,int b) { return a/b; } } class ExceptionDemoTest { public static void main(String[] args) { ExceptionDemo ed = new ExceptionDemo(); try { int x = ed.method(4,0); System.out.println("x = "+x); } catch (Exception e) { System.out.println(e.toString()); //输出异常的类的对象 System.out.println("除零了"); } finally { System.out.println("over"); } } }
第二种处理方式(用户的功能方法抛出异常,交给主方法抛出异常,最后由虚拟机JVM进行处理):
class ExceptionDemo { public int method(int a,int b) throws Exception { return a/b; } } class ExceptionDemoTest { public static void main(String[] args) throws Exception { ExceptionDemo ed = new ExceptionDemo(); ed.method(4,0); System.out.println("over"); } }
第三种处理方式(用户的功能方法抛出异常,接着由用户自己检测、捕获、处理):
class ExceptionDemo { public int method(int a,int b) throws ArithmeticException,ArrayIndexOutOfBoundsException { int[] array = new int[a]; System.out.println(array[4]); return a/b; } } class ExceptionDemoTest { public static void main(String[] args) { ExceptionDemo ed = new ExceptionDemo(); try { //int x = ed.method(4,0); //所传两个数据均发生了异常,即数组下标越界异常和除数为0异常,但这两个 //异常不能同时检测到,因为检测时有循序,但第一个检测到处理完时即停止 // int x = ed.method(4,1); //检测到数组越界异常 int x = ed.method(5,0); //检测到除0异常 System.out.println("x = "+x); } catch (ArithmeticException e) { System.out.println("异常的数据信息为"+e.getMessage()); System.out.println(e.toString()); //输出异常的类的对象 System.out.println("除零了"); //e.printStackTrace(); //其实它是虚拟机的处理机制 } catch (ArrayIndexOutOfBoundsException e) { System.out.println("异常的数据信息为"+e.getMessage()); System.out.println(e.toString()); //输出异常的类的对象 System.out.println("数组角标越界异常"); //e.printStackTrace(); //其实它是虚拟机的处理机制 } catch(Exception e) //这个是上面的子类异常的父类异常,多用于处理不在具体的异常之外的异常 { System.out.println(e.toString()); //输出异常的类的对象 } finally { System.out.println("over"); } } }
使用系统异常类Exception的子类来自定义异常
具体举例如下:
例子1:
需求:人为自定义异常(除数不能为负数)
//继承Exception的异常类
class FuShuException extends Exception { /* private String msg; FuShuException(String msg) { this.msg = msg; } public String getMessage() //输出异常信息 { return msg; } */ private int vaule; FuShuException(String msg,int vaule) { super(msg); //父类throwable已经完成了异常的处理,子类直接可以调用,所以上面的异常初始化方法不用再写 this.vaule = vaule; } public int getVaule() { return vaule; } }
//异常类 class ExceptionDemo { public int method(int a,int b) throws FuShuException { if(b<0) throw new FuShuException("出现了除数是负数的情况:/by fushu",b); //通过手动throw抛出自定义异常对象 return a/b; } }
//主方法测试 class ExceptionDemoTest { public static void main(String[] args) { ExceptionDemo ed = new ExceptionDemo(); try { int x = ed.method(4,-1); System.out.println("x = "+x); } catch (FuShuException e) { System.out.println(e.toString()); //输出自定义异常类,自动去调用前面的get Message(); System.out.println("错误的负数是:"+e.getVaule()); //System.out.println("除数为负数了"); } finally { System.out.println("over"); } } }
例子2:
异常的举例描述:老师用电脑讲课
(封装成对象)异常类:
1、电脑蓝屏(BuleScreenException),抛出后老师可以捕获解决。
2、电脑冒烟(SystemSmokingException),抛出后老师解决不了,可以转换为课程进度异常抛出,老师可以捕获解决 即产生第三个异常类NoPlanExecption
提炼需求类:
1、老师:成员变量(name)、成员方法(lecture)
2、电脑:成员方法(run、reset)
//继承Exception的电脑蓝屏异常类
class BlueScreenException extends Exception { private String message; public BlueScreenException(String message) { super(message); } }
//继承Exeption的系统冒烟异常类 class SystemSmokingException extends Exception { private String message; public SystemSmokingException(String message) { super(message); } }
//继承Exception的转换异常类 class NoPlanExecption extends Exception { private String message; public NoPlanExecption(String message) { super(message); } }
//定义一个老师类 class Teacher { private String name; private Computer c; public Teacher(String name) { this.name = name; c = new Computer(3); //设置电脑目前的状态(1或2或3) } void lecture() throws NoPlanExecption { try { c.run(); } catch (BlueScreenException e) { System.out.println(e.toString()); c.reset(); } catch(SystemSmokingException e) { test(); throw new NoPlanExecption("讲课不能继续,"+e.getMessage()); //冒烟异常老师解决不了,转换为课程进度异常抛出 } System.out.println(name+"讲课"); } public void test() { System.out.println("学生自己做练习"); } }
//定义一个电脑类 class Computer { private int state; //电脑状态:数字1代表正常运行、数字2代表蓝屏、数字3代表冒烟 public Computer(int state) { this.state = state; } void run() throws BlueScreenException,SystemSmokingException { if(state==2) throw new BlueScreenException("电脑蓝屏了"); else if(state==3) throw new SystemSmokingException("电脑冒烟了"); System.out.println("电脑运行"); } public void reset() { state = 1; //重新将电脑状态设置成数字1 System.out.println("电脑重新启动"); } }
//测试类 class CustomizeExceptionDemo { public static void main(String[] args) { Teacher t = new Teacher("毕老师"); try { t.lecture(); } catch (NoPlanExecption e) { System.out.println(e.toString()); System.out.println("换老师或放假"); } } }