异常分两种大的异常类型,运行时异常和受检查异常。
用户既可以使用系统的异常类来处理异常信息,也可以创建系统的异常类的子类来自定义异常,这种方式比较灵活,虚拟机可以报出自己设置的异常信息,清楚明白。
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(,);
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(,);
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[]);
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(,); //检测到除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<)
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(,-);
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(); //设置电脑目前的状态(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==)
throw new BlueScreenException("电脑蓝屏了"); else if(state==)
throw new SystemSmokingException("电脑冒烟了"); System.out.println("电脑运行");
}
public void reset()
{
state = ; //重新将电脑状态设置成数字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("换老师或放假");
}
}
}