------Java培训、Android培训、iOS培训、.Net培训、期待与您交流! -------
一、 Error和Exception
1. 一般不可处理的:Error
特点:是由jvm抛出的严重性问题。
这种问题发生,一般不针对性处理,直接修改程序。
2. 可以处理的:Exception 该体系的特点:
子类的后缀名都是用其父类名作为后缀,阅读性很强。
二、 Java中异常分为哪些种类
按照异常需要处理的时机分为编译时异常也叫CheckedException和运行时异常也叫RuntimeException。只有java语言提供了Checked异常,Java认为Checked异常都是可以被处理的异常,所以Java程序必须显式处理Checked异常。如果程序没有处理Checked异常,该程序在编译时就会发生错误无法编译。这体现了Java的设计哲学:没有完善错误处理的代码根本没有机会被执行。对Checked异常处理方法有两种:
1、当前方法知道如何处理该异常,则用try...catch块来处理该异常。
2、当前方法不知道如何处理,则在定义该方法是声明抛出该异常。
运行时异常只有当代码在运行时才发行的异常,编译时不需要try、catch。Runtime如除数是0和数组下标越界等,其产生频繁,处理麻烦,若显示申明或者捕获将会对程序的可读性和运行效率影响很大。所以由系统自动检测并将它们交给缺省的异常处理程序。当然如果你有处理要求也可以显示捕获它们。
三、 异常的使用:
Throwable中的方法:
1. getMessage():获取异常信息,返回字符串。
2. toString():获取异常类名和异常信息,返回字符串。
3. printStackTrace():获取异常类名和异常信息,以及异常出现在程序中的位置,返回值void。
4. printStackTrace(PrintStream s):通常用该方法将异常内容保存在日志文件中,以便查阅。
示例:
class Demo{
public static int method(int[] arr, int index){
if(arr == null)
{
// throw抛出异常
throw new NullPointerException("数组的引用不能为空!");
}
if(index >=arr.length ){
//自定义异常抛出时显示的信息
throw new ArrayIndexOutOfBoundsException("数组的角标越界:" + index);
}
return arr[index];
}
}
class ExceptionDemo{
public static void main(String[] args)//调用的函数是运行时异常,无需声明
{
int[] arr = newint[3];
Demo.method(arr,30);
}
}
四、 自定义异常
1、可以自定义出的问题称为自定义异常。
如上例中对于角标越界的情况java中有定义过,但是对于针对角标为负数的情况,可以用负数角标异常来表示,负数角标这种异常在java中并没有定义过。按照java异常的创建思想,面向对象,将负数角标进行自定义描述,并封装成对象。
这种自定义的问题描述称为自定义异常。
2、使用自定义异常时也是通过throw将自定义异常抛出。需要注意的是,throws和throw的区别:
!. throws用于标识函数暴露出的异常类,并且可以抛出多个,用逗号分隔。throw用于抛出异常对象。
!!. thorws用在函数上,后面跟异常类名。throw用在函数内,后面跟异常对象。
!!!.定义功能方法时,需要把出现的问题暴露出来让调用者去处理,那么就通过throws在函数上标识。
!!!!.在功能方法内部出现某种情况,程序不能继续运行,需要进行跳转时,就用throw把异常对象抛出。
示例:
class FuShuIndexException extends Exception//继承Exception该类为异常类
{
FuShuIndexException(){}
FuShuIndexException(Stringmsg){
super(msg);
}
}
class Demo{
public static int method(int[]arr, int index) throws FuShuIndexException//标致该方法抛出FuShuIndexException异常
{
if(index < 0)
{
//只有继承了Exception才具备throw方法
throw new FuShuIndexException("数组的角标是负数啦!" );
}
return arr[index];
}
}
class ExceptionDemo{
//调用的方法存在FuShuIndexException异常,这里也需要标志
public static voidmain(String[] args) throws FuShuIndexException
{
int[]arr = new int[3];
Demo.method(arr,-30);
}
}
3、自定义RuntimeException异常
Exception和其子类都是编译时被检测异常,这种问题一旦出现,希望在编译时就进行检测,让这种问题有对应的处理方式。而Exception中的RuntimeException和其子类是编译时不检测异常即运行时异常,这种问题的发生,无法让功能继续,运算无法运行,更多是因为调用的原因导致的或者引发了内部状态的改变导致的。这种问题一般不处理,直接编译通过,在运行时,让调用者调用时的程序强制停止,让调用者对代码进行调整。
上例其实可以定义为RuntimeException异常,在编译时可以通过,在运行时抛出:
代码:
class FuShuIndexException extends RuntimeException{
FuShuIndexException(){}
FuShuIndexException(Stringmsg){
super(msg);
}
}
class Demo{
public static int method(int[]arr, int index){//RuntimeException没有必要用throws抛出,并不是必须要处理
if(index < 0){
throw new FuShuIndexException("数组的角标是负数啦!" );
}
return arr[index];
}
}
五、 异常的处理
1、具体格式是:
try{
//需要被检测异常的代码。
}
catch(异常类 变量) //该变量用于接收发生的异常对象
{
//处理异常的代码。
}
finally{
//一定会执行的代码;
}
注意:finally代码块只有一种情况不会被执行,就是在之前执行了System.exit(0)。
2、处理过程:
try中检测到异常会将异常对象传递给catch,catch捕获到异常进行处理。
finally里通常用来关闭资源。比如:数据库资源,IO资源等。
需要注意:try是一个独立的代码块,在其中定义的变量只在该变量块中有效。
如果在try以外继续使用,需要在try外建立引用,在try中对其进行初始化。IO,Socket就会遇到。
异常处理的原则:
!. 函数内容如果抛出需要检测的异常,那么函数上必须要声明。 否则,必须在函数内用try/catch捕捉,否则编译失败。
!!. 如果调用到了声明异常的函数,要么try/catch,要么throws,否则编译失败。
!!!. 什么时候catch,什么时候throws呢? 功能内容可以解决,用catch。 解决不了,用throws告诉调用者,由调用者解决。
!!!!. 一个功能如果抛出了多个异常,那么调用时,必须有对应多个catch进行针对性处理。内部有几个需要检测的异常,就抛几个异常,抛出几个,就catch几个。
示例:
class Demo{
public int show(int index)throws ArrayIndexOutOfBoundsException{
if(index < 0)
throw newArrayIndexOutOfBoundsException("越界啦!");
int[] arr = new int[3];
return arr[index];
}
}
class ExceptionDemo{
public static voidmain(String[] args){
Demod = new Demo();
try{
int num = d.show(-3);
System.out.println("num = " + num);
}catch(ArrayIndexOutOfBoundsException e){
System.out.println(e.toString());
System.exit(0);//退出jvm
}finally{//通常用于关闭(释放)资源
System.out.println("finally");//由于前面执行了System.exit(0);,故不会执行此语句。
}
System.out.println("over");
}
}