------- android培训、java培训、期待与您交流! ----------
一、异常的概念
1、异常:程序在运行时出现不正常情况。对于问题的划分有两种:一种是严重的问题,一种是非严重的问题:
无论Error或者Exception都具有一些共性内容。 比如:不正常情况的情况,引发原因等。
|---Error
|---Exception
2、异常分为两种:
1)、编译时被检测的异常;
2)、编译时不被检测的异常(运行时异常: RuntimeException及其子类)。
3、异常的好处:
1)、可以将问题进行封装;
2)、将正常流程代码和问题处理代码相分离,便于阅读。
二、异常的的处理
1、异常的处理方式
Java提供了特殊的语句对异常进行处理。
try
{
需要被检测的代码;
}
catch(异常类 变量)
{
处理异常的代码(处理方式);
}
finally
{
一定会执行的代码;
}
finally代码块:定义一定会执行的代码,通常用于关闭资源。只有一种情况finally执行不到:System.exit(0); 即系统退出,Jvm停止。
注:catch是用于处理异常,如果没有catch就代表异常没有被处理过,如果该异常是检测异常,那么必须声明。
2、对捕获到的异常对象进行常见操作
StringgetMessage(): 获取异常的信息;
toString(): 获取异常名称,异常信息;
printStackTrace(): 获取异常名称,异常信息,异常出现的位置。
演示示例:
class Demo
{
public int div(int a,int b) throws ArithmeticException,ArrayIndexOutOfBoundsException
{
int[] arr = new int[a];
System.out.println(arr[4]);
return a/b;
}
}
class ExceptionDemo
{
public static void main(String[] args)
{
Demo d = new Demo();
try
{
int x = d.div(5,0);
System.out.println("x= "+x);
}
catch (ArithmeticException e) //此处想到于Exception e = new ArithmeticException;
{
System.out.println("出现除数为零错误!!!");
System.out.println(e.getMessage()); //获取异常信息;
System.out.println(e.toString()); //获取异常名称,异常信息;
e.printStackTrace(); //获取异常名称,异常信息,异常出现的位置;
}
catch (ArrayIndexOutOfBoundsException e)
{
System.out.println("出现角标越界错误!!!");
System.out.println(e.getMessage());
System.out.println(e.toString());
e.printStackTrace();
}
System.out.println("over");
}
}
其实
java对异常的处理机制就是在调用printStackTrace方法,打印异常在堆栈中的跟踪信息。
在函数上声明异常,便于提高安全性,让调出进行处理,不处理编译失败。
3、对多异常的处理:
1)、声明异常时,建议声明更为具体的异常,这样可以更加具体的处理异常;
2)、对方声明几个异常,就对应几个catch块,不要定义多余的catch块。如果多个catch块中出现继承关系,父类(Exception)异常catch块放在最下面。(如果放在上面,有的子类异常catch块将执行不到);
3)、建议在catch处理时,catch中一定要定义具体处理方式,不要简单一句e.printStackTrace(),也不要简单地就写一条输出语句。
4、自定义异常
因为项目中会出现一些特有的问题,而这些问题并没有被java所描述并封装,所以对这些特有的问题可以按照java对问题封装的思想将这些特有的问题进行自定义的异常封装。
当在函数内部出现了throw抛出异常对象,那么就必须要给出对应的处理动作,要么在内部try catch处理,要么在函数上声明让调用者处理。
一般情况下,函数内出现异常,函数需要声明。如果发现打印的结果中只有异常名称,没有异常信息,是因为自定义的异常并未定义所属信息。
5、如何自定义异常
因为父类中已经把异常信息的操作都完成了,所以子类只要在构造时将异常信息通过super传递给父类,那么就可以直接通过getMessage方法获取自定义异常信息。
演示示例:
class FuShuException extends Exception //自定义负数异常;继承Exception
{
private int value;
FuShuException(String message,int value)
{
super(message); //将异常信息传递给父类;
this.value = value;
}
public int getValue()
{
return value;
}
}
class Demo
{
public int div(int a,int b) throws FuShuException //抛出负数异常
{
if(b<0)
throw new FuShuException("除数出现负数了",b);
return a/b;
}
}
class ExceptionDemo1
{
public static void main(String[] args)
{
Demo d = new Demo();
try
{
int x = d.div(4,-3);
System.out.println("x = "+x);
}
catch (FuShuException e)
{
System.out.println(e.toString()); //通过父类获取异常信息;
System.out.println("出现的负数是:"+e.getValue());
}
System.out.println("over");
}
}
throw和throws的区别:
1)、throws使用在函数上。而throw使用在函数内;
2)、throws后面跟的是异常类类,还可以跟多个,中间用逗号隔开。而throw后面跟的是异常对象。
例如:
class Demo
{
int div(int a, int b)throws FuShuException
{
If(b<0)
throw new FuShuException(“出现除以负数的情况了”);
}
}
三、RuntimeException
1、Exception中有一个特殊的子类异常RuntimeException:运行时异常。
其特点为:
1)、如果在函数内中抛出该异常,函数上可以不用声明,编译照常通过;
2)、 如果在函数上声明了该异常,调用者可以不进行处理,编译照常通过。
原因:之所以不用在函数上声明,是因为不需要让调用者处理。当该异常发生时希望程序停止。因为在运行时出现了无法继续运算的情况,希望程序停止后对代码进行修正。
自定义异常时,如果该异常导致程序无法再继续运行,就让自定义异常继承RuntimeException。
演示示例:
abstract class Shape
{
abstract void getArea();
}
class NoValueException extends RuntimeException //继承RuntimeException是因为当半径为负值后
{ //无法继续运算,让程序停止,修正代码
NoValueException(String message)
{
super(message);
}
}
class Circle extends Shape
{
double radius;
public static final double PI=3.14;
Circle(double radius)
{
if(radius <= 0)
throw new NoValueException("半径不能为负数"); //抛出RuntimeException的子类异常;
this.radius = radius; //方法上不用声明;后续也不用处理
}
void getArea()
{
System.out.println(radius*radius*PI);
}
}
class ExceptionDemo2
{
public static void main(String[] args)
{
Circle c = new Circle(-5.0);
c.getArea();
System.out.println("over");
}
}
1、子类在覆盖父类时,如果父类的方法抛出异常,那么在子类的覆盖方法中,只能抛出父类异常或者父类异常的子类;
2、如果父类方法抛出多个异常,那么子类在覆盖该方法时,只能抛出父类异常的子集;
3、如果父类或者接口的方法中没有异常抛出,那么子类在覆盖方法时,也不可以抛出异常。如果子类方法中发生了异常,就必须进行try处理,绝对不能抛出。
五、异常处理的原则
1、处理方式有两种:try或者throw;
2、调用到抛出异常的功能时,抛出几个就处理几个,一个try对应多个catch;
3、当出现多个catch,父类的catch放在最下面;
4、Catch内需要定义针对性的处理方式,不要简单的定义printStackTrace输出语句,也不要不写。
当捕获到的异常本功能处理不了时,可以继续在catch中抛出。
try
{
thrownew AException();
}
catch (AException e)
{
throwe;
}
如果异常处理不了,但并不属于该功能出现的异常,可以将异常转换后再抛出和该功能相关的异常。或者异常可以处理,但需要将异常产生的和本功能相关的问题提供出去,让调用者知道并处理,也可以将捕获的异常处理后,转换新的异常。
try
{
thrownew AException();
}
catch (AException e)
{
//对AException处理;
thrownew BException();
}