Java:Exception
异常:
就是程序在运行时出现不正常的情况。
异常的由来:问题也是现实生活中一个具体的事物,也可以通过java的类的形式进行描述,并封装成对象。其实就是Java对不正常情况进行描述后的对象的体现。
两种问题:
严重问题(Error)、非严重问题(Exception)。
Error和Exception具有一些共性内容。(都是Throwable的子类)
try { 需要被检测的代码 } catch(异常类 变量) { 处理异常的代码;(处理方式) }
严重问题:
Java通过Error类进行描述。对于Error一般不编写针对性的代码对其进行处理。
非严重问题:
Java通过Exception类进行描述。对Exception可以使用针对性的代码进行处理。
异常的处理:
try { 需要被检测的代码 } catch(异常类 变量) { 处理异常的代码;(处理方式) }
异常的常见操作:
getMessage() :返回信息:异常原因
toString():反馈信息:异常类+异常原因
printStackTrace():反馈堆栈中的异常信息 异常类+异常原因+异常出现的位置,其实jvm默认的异常处理机制就是在调用printStackTrace方法
例如,除数是零
class Demo { int div(int a, int b) { return a/b; } } class ExceptionDemo { public static void main(String[] args) { Demo d = new Demo(); try { int x = d.div(4,0);//捕获到异常对象 new AritchmeticException() System.out.println("x=" + x); } catch (Exception e)//把捕获到的异常给了catch: Exception e = new AritchmeticException(); { System.out.println("除数是零!"); System.out.println(e.getMessage()); System.out.println(e.toString()); e.printStackTrace(); } System.out.println("Program Over!"); } }
但我们在定义Class时就应该提示是否会出现异常,提醒调用该Class时是需要处理Exception。当调用者使用该Class时,必须捕获该类的异常进行处理或者抛出。
捕获:try-catch处理
class Demo { int div(int a, int b)throws Exception//在功能上通过throws关键字声明了该功能在使用时有可能出现问题 { ... } } class ExceptionDemo { public static void main(String[] args) { Demo d = new Demo(); try { ... } catch (Exception e) { ... } } }抛出:调用该有标识异常的类的方法上通过throws抛出,不处理
class Demo { int div(int a, int b)throws Exception//在功能上通过throws关键字声明了该功能在使用时有可能出现问题 { ... } } class ExceptionDemo { public static void main(String[] args)throws Exception { ... } }
·声明异常时,建议声明更为具体的异常,这样处理的可以更具体。
例如上面的例子,可以把异常由Exception更改为ArithmeticExcption
·有多个异常时,可以针对性用多个catch处理。如果多个异常有继承关系,父类异常的catch放在最下面
建议进行catch处理时,catch中一定要定义具体的处理方式,不要简单的使用printStackTrace()等等
class Demo { int div(int a, int b)throws ArithmeticException,ArrayIndexOutOfBoundsException//抛出多个异常 { int[] arr = new int[a]; System.out.println(arr[5]);//可能出现异常的地方 return a/b;//可能出现异常的地方 } } class ExceptionDemo { public static void main(String[] args) { Demo d = new Demo(); try { int x = d.div(4,0); System.out.println("x=" + x); } catch (ArithmeticException e)//处理异常 { System.out.println("除数是零!"); e.printStackTrace(); } catch (ArrayIndexOutOfBoundsException e)//处理异常 { System.out.println("数组角标异常!"); e.printStackTrace(); } System.out.println("Program Over!"); } }
自定义异常:
因为项目中会出现特有问题,而这些问题是Java未对其描述并封装成对象。所以对这些特有问题可以按照Java对问题的封装思想,对其封装。
当在函数内部出现了throw抛出异常对象,那么就必须要给出对应的处理。要么在内部try-catch,要么在函数上声明让调用者处理。 一般情况下,函数内出现异常,函数上需要声明。
如何自定义异常信息呢?
因为父类已经把异常信息的操作都完成了,所以子类只要在构造时,将异常信息传递给父类,通过super语句。那么就可以直接通过getMessage方法获取自定义的异常信息了。
继承Exception的原因
异常体系都有一个特点:异常类和异常对象都被抛出。
他们都具有可抛性,是Throwable这个体系的独有特点。只有这个体系中的类和对象才可以被throws和throw操作。
throws 和 throw 的区别:
1,throws使用在函数上,throw使用在函数内
2,throws后面跟的是异常类,可以跟多个,用逗号隔开;throw后跟的是异常对象
1 /* 2 需求:在本程序中,不允许除数为负数。 3 因为该需求是特殊要求,所以要对这个问题进行自定义的描述; 4 */ 5 class DefinedException extends Exception 6 { 7 private int value; 8 9 DefinedException() 10 { 11 System.out.println("自定义异常的空参数的构造函数:除数不能为负数"); 12 } 13 DefinedException(String msg) 14 { 15 super(msg); 16 } 17 DefinedException(String msg, int value) 18 { 19 super(msg); 20 this.value = value; 21 } 22 23 public int getValue() 24 { 25 return value; 26 } 27 } 28 29 class Demo 30 { 31 int div(int a,int b)throws DefinedException 32 { 33 if (b<0) 34 { 35 //throw new DefinedException(); 36 //throw new DefinedException("除数不能为负数!"); 37 throw new DefinedException("除数不能为负数!",b);//手动通过throw抛出 38 } 39 return a/b; 40 } 41 } 42 class DefinedExceptionDemo 43 { 44 public static void main(String[] args) 45 { 46 Demo d = new Demo(); 47 try 48 { 49 int x = d.div(4,-1); 50 System.out.println("a/b="+x); 51 } 52 catch (DefinedException e) 53 { 54 System.out.println("除数不能为负数!"); 55 System.out.println("错误的数是:"+e.getValue()); 56 //d.printStackTrace(); 57 } 58 59 System.out.println("Program Over!"); 60 } 61 62 }
RuntimeException:
Exception中有一个特殊的子类异常RuntimeException 运行时异常。
RuntimeException或者其子类的异常,可以不用在函数上使用throws申明。如果在函数上申明了该异常,调用者可以不用处理就可以通过编译。不申明的原因:当程序运行时遇到该异常,导致下面的程序不能预期运行,希望停止运行后修正该程序。(比如如果一个函数只申明了一个异常,而在函数内抛出了三个异常,那么另外两个异常必然就是RuntimeException或者其子类)
自定义异常时,如果该异常发生后无法继续工作,就让该异常继承RuntimeException
异常分两种:
1,编译时被检测的异常
2,编译时不被检测的异常,但运行时异常。RuntimeException及其子类
finally:
finally中的代码是一定会执行的代码,一般存放必须要操作的内容比如释放与数据库的连接、关闭IO等等。
finally只有一种情况不会被执行到:finally前面有System.exit(0)
异常处理的三种各式:
1, try-catch
2, try-catch-finally
3, try-fiinally
异常在子父类覆盖时的体现:
1,子类在覆盖父类方法时,如果父类的方法抛出异常,那么子类的覆盖方法只能抛出父类的异常或者该异常的子类
2,如果父类方法抛出多个异常,那么子类在覆盖该方法时,只能抛出父类异常的子集
3,如果父类或者接口的方法中,没有异常抛出,那么子类在覆盖方法时也不可以抛出异常
如果子类方法发生了异常,就必须要try处理,绝对不能抛。