Java中的异常

来源:互联网 发布:淘宝 英雄杀cdk 编辑:程序博客网 时间:2024/05/17 08:51

一、什么是异常

异常(Exception)定义了程序中遇到的非致命的错误, 而不是编译时的语法错误。如果程序中有异常而没有处理,一般可以通过编译,但运行时会发生错误。

先看下面这个例子:

class A{public int divide(int a, int b){int m = 0;m = a/b;return m;}}public class TestExcept{public static void main(String[] args){new A().divide(6, 0);                //除数为0,可以通过编译运行会报错System.out.printf("测试异常\n");    //15行抛出异常,该语句不会运行}}
运行的结果为:

-------------------------------------------------------------------------------------------------

Exception in thread "main" java.lang.ArithmeticException: / by zero
        at A.divide(TestExcept.java:6)
        at TestExcept.main(TestExcept.java:15)

-------------------------------------------------------------------------------------------------

上面的程序运行的结果报告发生了算术异常(ArithMethicException),系统不再执行下去,提前结束,这种情况就是我们所说的异常。


二、异常的分类


1、Error是系统错误,编译时就会报错,程序员无法处理

2、Exception是程序员可以捕获并处理的异常

3、RuntimeException的子类异常时可以处理也可以不处理的异常

4、凡是继承自Exception但又不是RuntimeException子类的异常,必须进行捕获和处理


三、异常处理机制

当Java程序运行发生异常时,系统会自动检测到该错误,并立即生成一个与该错误对应的异常对象。然后把该异常对象提交给Java虚拟机,Java虚拟机会自动寻找相应的处理代码来处理这个异常,如果没有找到,则程序终止。程序员可以自己编写代码来捕获可能出现的异常,并编写代码来处理相应的异常,其中可以用来处理异常的一种方式便是try..catch语句。

try..catch语句的用法:

try {      // 可能会发生异常的程序代码  } catch(ExceptionName1 id1){      // 当产生ExceptionName1异常时的处理措施 }  catch(ExceptionName2 id2){       //当产生ExceptionName2异常时的处理措施  } ....finally{//无论是否捕获到异常都必须处理的代码}
注意:

1、所有的catch只能有一个被执行

2、有可能所有的catch都没有执行,但finally一定会执行

3、要先catch子类异常在catch父类异常,否则会报错

4、当try 代码块中的语句发生了异常,程序就会跳转到catch 代码块中执行,执行完catch 代码块中的程序代码后,系统会继续执行catch 代码块后的其他代码,但不会执行try 代码块中发生异常语句后的代码

例子:

class A{public int divide(int a, int b){int m = 0;m = a/b;return m;}}public class TestExcept_1{public static void main(String[] args){A aa = new A();try{aa.divide(6,0);System.out.printf("测试异常1\n");     //该语句不会运行}catch(ArithmeticException e){System.out.printf("测试异常2\n");}catch(NullPointerException e){System.out.printf("测试异常3\n");}catch (Exception e)  //不能写在前面,否则会报错,因为Exception是ArithmeticException和NullPointerException的父类{System.out.printf("测试异常4\n");}finally{System.out.printf("测试异常5\n");}}}
运行结果为:

---------------------------------------------------------------------------------------------------------

测试异常2
测试异常5

----------------------------------------------------------------------------------------------------------
分析:

当try 代码块中的程序发生了异常,系统将这个异常发生的代码行号,类别等信息封装到一个对象中,并将这个对象传递给catch 代码块。catch 关键字后跟有一个用括号括起来的Exception 类型的参数e,这跟我们经常用到的如何定义一个函数接收的参数格式是一样的。括号中的Exception 就是try 代码块传递给catch 代码块的变量类型,e 就是变量名,所以我们也可以将e 改用成别的名称(如ex 等)


四、throws关键字

针对上面的例子,我们假设TestExcept_1类与A类不是同一个人写的,写TestExcept_1类的人,在main 方法中调用A类的devide 方法时,怎么能知道devide 方法有可能出现异常情况呢?他又怎么能够想到要用try catch 语句去处理可能发生的异常呢?

问题可以这样解决,只要写A类的人,在定义devide 方法时,在devide 方法参数列表后用throws 关键字声明一下,该函数有可能发生异常及异常的类别。这样,调用者在调用该方法时,就必须用try…catch 语句进行处理,否则,编译将无法通过。

如下面的程序代码:

class A{public int divide(int a, int b) throws ArithmeticException{int m = 0;m = a/b;return m;}}public class TestExcept_2{public static void main(String[] args){new A().divide(6, 0);                //调用了divide方法但没有进行处理,程序运行会报错System.out.printf("测试异常\n");   }}
运行结果为:

------------------------------------------------------------------------------------------------------------------------------------------------

Exception in thread "main" java.lang.ArithmeticException: / by zero
        at A.divide(TestExcept.java:6)
        at TestExcept.main(TestExcept.java:15)

------------------------------------------------------------------------------------------------------------------------------------------------

注意:

void f() throws A{....}

1、throws A表示调用f方法时可能会抛出A类异常,建议调用f方法时做好对f方法可能抛出的A类异常进行捕获

2、throws A并代表f方法一定会抛出A类异常

3、throws A 不代表调用f方法是,必须对A异常进行捕获。假如A是RuntimeException子类异常,则编译器允许可以不进行处理

建议:

1、对throws出的所有异常进行处理

2、如果一个方法内部已经对A异常进行了处理。则就不要再throws A


五、自定义异常

除了系统提供的异常,我们也可以定义自己的异常类,自定义的异常类必须继承Exception 类。

throw关键字:

throw用来抛出异常,其格式为:

throw new 异常名(参数);

具体怎么自定义异常看下面这个例子:

class XxxException extends Exception     //自定义异常XxxException{public XxxException(String name){super(name);                    //继承父类的构造方法}}class YyyException extends Exception    //自定义异常XxxException{public YyyException(String name){super(name);}}class A {public int x;public A(int x){this.x = x;}public void fun() throws XxxException, YyyException{if(x==0)throw new YyyException("Yyy");  //抛出异常if(x==1)throw new XxxException("Xxx");   //抛出异常}}public class TestExcep_2{public static void main(String[] args){A aa = new A(0);try{aa.fun();}catch(YyyException e){System.out.printf("测试异常1\n");e.printStackTrace();            //打印出异常的具体信息}catch(XxxException e){System.out.printf("测试异常2\n");e.printStackTrace();}catch (Exception e){System.out.printf("测试异常3\n");e.printStackTrace();}finally{System.out.printf("测试异常4\n");}}}
输出结果:

-----------------------------------------------------------------------------------------------------------------------------------------------------------------

测试异常1
YyyException: Yyy
        at A.fun(TestExcep_2.java:25)
        at TestExcep_2.main(TestExcep_2.java:39)
测试异常4

------------------------------------------------------------------------------------------------------------------------------------------------------------------


0 0