------- android培训、java培训、期待与您交流! ----------
异常
异常:就是程序在运行时出现不正常情况
异常由来:问题也是现实生活中的一个具体的事物,也可以通过Java的类的形式进行描述。并封装成对象。其实就是Java对不正常情况进行描述后的对象体现。
对于问题的划分:两种:一种是严重的问题,一种非严重的问题
对于严重的,Java通过Error类进行描述。对于Error一般不编写针对性的代码对其进行处理
对于非严重的,Java通过Exception类进行描述。对于Exception可以使用针对性的处理方式进行处理。
无论Error或者Exception都具有一些共性内容。比如:不正常情况的信息,引发原因等。
异常的体系:
Throwable
|--Error
|--Exception
|--RuntimeException
异常的特点:异常体系中所有类以及建立的对象都具有可抛性。也就是说可以被throw和throws关键词所操作
异常的用法:throw定义在函数内,用于抛出异常对象
throws定义在函数上,用于抛出异常类,可以抛出多个用逗号隔开
Java提供了特有的语句进行处理
try
{
需要被检测的代码;
}
catch(异常类 变量)
{
处理异常的代码:(处理方式);
}
finally
{
一定会执行的语句;
}
注意:1、finally中定义的通常是关闭资源的代码。因为资源必须释放
2、finally只有一种情况下不会执行。当执行到System.exit(0),finally不会执行。
当函数内容有throw抛出异常类对象,并未进行try处理。必须在函数上声明,否则编译失败。
注意 :RuntimeException除外。函数内抛出该异常时,可以不用声明。
对捕获到的异常对象进行常见方法操作
String getMessage():获取异常信息
在函数上声明异常。
便于提高安全性,让调用处进行处理。不处理编译失败。
对多异常的处理
1、声明异常时,建议声明更为具体的异常。这样处理的可以更具体。
2、对方声明几个异常,就对应有几个catch块,不要定义多余的catch块。
如果多个catch块中的异常出现继承关系,父类异常通常放在最下面。
class Demo
{
int div(int a,int b)throws ArithmeticException,ArrayIndexOutOfBoundsException//在功能上通过throws的关键字声明了该功能有可能会出现问题。
{
int[] arr = new int[a];
System.out.println(arr[4]);
return a/b;
}
}
class ExceptionDemo2
{
public static void main(String[] args) //throws Exception
{
Demo d = new Demo();
try
{
int x = d.div(5,0);
System.out.println("x="+x);
}
catch (ArithmeticException e)
{
System.out.println(e.toString());
System.out.println("被零除了!!");
}
catch (ArrayIndexOutOfBoundsException e)
{
System.out.println(e.toString());
System.out.println("角标越界啦!!");
}
catch(Exception e)//父类异常一般放在最后面
{
System.out.println("hahah:"+e.toString());
}
System.out.println("over");
}
}
输出:0
java.lang.ArithmeticException: / by zero
被零除了!!
over
自定义异常:
原因:因为项目中会出项特有的问题
而这些问题并未被Java所描述并封装对象。
所以对于这些特有的问题可以按照Java的对问题的封装思想。
将特有的问题进行自定义的异常封装
如何定义异常信息呢?
因为父类中已经把异常信息的操作都完成了。所以子类在构造时,将异常信息传递给父类通过super语句。那么就直接可以通过getMessage方法获得自定义的异常信息。
所以自定义异常,必须自定义继承Exception
继承Exception原因:
异常体系有一个特点:因为异常类和异常对象都被抛出
他们都具备可抛性。这个可抛性是Throwable这个体系中独有的特点。
因为只有这个体系中的类和对象可以被throw和throws操作。
那么throw和throw的区别:
throws使用在函数上;throw使用在函数内;
class FuShuException extends Exception //getMessage();
{
private int value;
FuShuException()
{
super();
}
FuShuException(String msg,int value)
{
super(msg);
this.value = value;
}
public int getValue()
{
return value;
}
}
class Demo
{
int div(int a,int b)throws FuShuException
{
if(b<0)
throw new FuShuException("出现了除数是负数的情况------ / by fushu",b);//手动通过throw关键字抛出一个自定义异常对象。
return a/b;
}
}
class ExceptionDemo3
{
public static void main(String[] args)
{
Demo d = new Demo();
try
{
int x = d.div(4,-9);
System.out.println("x="+x);
}
catch (FuShuException e)
{
System.out.println(e.toString());
//System.out.println("除数出现负数了");
System.out.println("错误的负数是:"+e.getValue());
}
System.out.println("over");
}
}
输出:FuShuException: 出现了除数是负数的情况------ / by fushu错误的负数是:-9
over
Exception中有一个特殊的子类异常RuntimeException:运行时异常
如果函数内容抛出该异常,函数上可以不声明,编译一样通过。
如果在函数上声明了该异常。调用者可以不用进行处理,编译一样通过。
之所以不用在函数上声明,是因为不需要让调用者处理。
当该异常发生,希望程序停止。因为在运行时,出现了无法继续运算的情况,希望停止程序后,对代码修正。
所以,自定义异常时,如果该异常的发生,无法在继续进行运算,就让自定义异常继承RuntimeException
从编译运行的角度异常有两种:
1、编译异常:异常在编译时,如果没有处理(未抛未try),编译失败;异常被标识,代表可以处理
2、运行异常:(编译时不检测)建议不处理,让程序停止,需对代码修正(RuntimeException以及其子类)
class FuShuException extends RuntimeException
{
FuShuException(String msg)
{
super(msg);
}
}
class Demo
{
int div(int a,int b)throws Exception//throws ArithmeticException
{
if(b<0)
throw new Exception("出现了除数为负数了");
if(b==0)
throw new ArithmeticException("被零除啦");
return a/b;
}
}
class ExceptionDemo4
{
public static void main(String[] args) throws Exception
{
Demo d = new Demo();
int x = d.div(4,-9);
System.out.println("x="+x);
System.out.println("over");
}
}
输出:Exception in thread "main" java.lang.Exception: 出现了除数为负数了
at Demo.div(ExceptionDemo4.java:14)
at ExceptionDemo4.main(ExceptionDemo4.java:28)
异常在子父类覆盖中的体现;
1,子类在覆盖父类时,如果父类的方法抛出异常,那么子类的覆盖方法,只能抛出父类的异常或者该异常的子类。
2,如果父类方法抛出多个异常,那么子类在覆盖该方法时,只能抛出父类异常的子集。
3,如果父类或者接口的方法中没有异常抛出,那么子类在覆盖方法时,也不可以抛出异常。
如果子类方法发生了异常。就必须要进行try处理。绝对不能抛。