java-异常进阶-包的使用
一 finally
1.1 异常执行的顺序
package test; public class Test { public static void main(String[] args) { Demo d = new Demo(); try { d.show(-5); System.out.println("hello try"); } catch (NoShowException e) { e.printStackTrace(); System.out.println("hello catch"); } } } class NoShowException extends Exception { NoShowException(String message) { super(message); } } class Demo { void show(int num)throws NoShowException { if(num<0) throw new NoShowException(num+",数值是非法的"); System.out.println("show ....."+num); } }
结果:
1.2 finally 作用
有一些特定的代码无论异常是否发生,都需要执行。
因为异常会引发程序跳转,导致有写语句执行不到。无法满足这个需求。
异常捕获处理时java提供解决方案。
应用场景;
定义一个功能往数据库中添加数据。 void add(Data data)throws NoAddException { //1,连接数据库。 try{ //2,添加数据。//添加数据时发生了异常情况。throw new SQLException();程序跳转,就执行不到断开连接。 //而断开连接必须要执行,因为不执行,连接资源在浪费。 //无论是否发生问题,都需要执行断开连接的动作,从而释放资源。 }catch(SQLException e) { //解决数据库的问题。 //同时将问题告诉调用者。 throw new NoAddException(); } finally { //3,断开连接。 } }
总结:finally到底什么时候用?
只要程序中使用到了具体的资源(数据库连接,IO资源,网络连接socket等)
需要释放,都必须定义在finally中。你在定义程序,只要问题发生与否,
指定程序都需要执行时,就定义finally中。
二 异常的组合方式
try catch finally 组合方式:
1 .try catch : 对代码进行异常检测,并对检测的异常传递给catch处理。
try catch : 对代码进行异常检测,并对检测的异常传递给catch处理。 异常捕获处理。 void show()//不用throws { try{ throw new Exception(); }catch(Exception e) { } }
2. try finally : 对代码进行异常检测,检测到异常后因为没有catch,所以一样会被默认jvm抛出。
异常是没有捕获处理的。但是功能所开启资源需要进行关闭,所有finally。
只为关闭资源。
void show()//需要throws { try{ throw new Exception(); }finally { } }
3,
try catch finally
检测异常,并传递给catch处理,并定义资源释放。
4,try catch1 catch2 catch3......
class ExceptionDemo10 { public static void main(String[] args) { System.out.println("Hello World!"); } }
三 异常在继承或实现的细节
异常在继承或者实现中的使用细节:★★★★★
1,子类覆盖父类方法时,如果父类的方法声明异常,子类只能声明父类异常或者该异常的子类,或者不声明。
2,当父类方法声明多个异常时,子类覆盖时只能声明多个异常的子集。
3,当被覆盖的方法没有异常声明时,子类覆盖时时无法声明异常的。
举例:父类存在这种情况,接口也有这种情况,
问题:接口中没有声明异常,而实现的子类覆盖方法时发生了异常,怎么办?
无法进行throws声明,只能catch的捕获。万一问题处理不了呢?catch中继续throw抛出,但是只能将异常转换成RuntimeException子类抛出。
1,子类覆盖父类方法时,如果父类的方法声明异常,子类只能声明父类异常或者该异常的子类,或者不声明。
原因:如是不是这样,会发生类型转换异常.具体看下面代码讲解
下面红色部分产生类型转换异常 //AException ex = new BException();
ps:父类有问题了,在覆盖的时候子类不能有更多的问题
Exception |--AException |--AAException |--BException class AException extends Exception { } class BException extends Exception { } class AAException extends AException { } class Fu { void show() { } } class Tool { void method(Fu f)//Fu f = new Zi(); { try { f.show(); } catch (AException ex)//AException ex = new BException(); { } } } Tool t = new Tool(); //t.method(new Fu()); t.method(new Zi()); class Zi extends Fu { void show()throws BException
{ } }
四 习题
1. 写出程序结果 class Demo { public static void func()//throws Exception { try { throw new Exception();//抛出编译时异常,并没有catch处理,必须throws声明。 } finally { System.out.println("B"); } } public static void main(String[] args) { try { func(); System.out.println("A"); } catch(Exception e) { System.out.println("C"); } System.out.println("D"); } } -----------B C D 2.写出程序结果 class Demo { public static void main(String[] args) { try { showExce(); System.out.println("A"); } catch(Exception e) { System.out.println("B"); } finally { System.out.println("C"); } System.out.println("D"); } public static void showExce()throws Exception { throw new Exception(); } } B C D 3.写出程序结果: class Demo { public static void func() { try { throw new Exception();//有throw抛出异常,功能会在这里结束。下面的语句都是废话。 System.out.println("A"); } catch(Exception e) { System.out.println("B"); } } public static void main(String[] args) { try { func(); } catch(Exception e) { System.out.println("C"); } System.out.println("D"); } }
B D ---------------- 4.写出程序结果 class Exc0 extends Exception{} class Exc1 extends Exc0{} class Demo { public static void main(String[] args) { try { throw new Exc1(); } catch(Exception e)//父类的catch必须放在最下面。否则编译失败。 { System.out.println("Exception"); } catch(Exc0 e) { System.out.println("Exc0"); } } }
--------------- 5.写出程序结果 @ class Test { public static String output=""; public static void foo(int i) { try { if(i==1) throw new Exception(); output+="1"; } catch(Exception e) { output+="2"; return; } finally { output+="3"; } output+="4"; } public static void main(String args[]) { foo(0); System.out.println(output);//134 foo(1); System.out.println(output); //13423 } }
134 13423 ---------------------- 6.编程题。 建立一个图形接口,声明一个面积函数。圆形和矩形都实现这个接口,并得出两个图形的面积。 注:体现面向对象的特征,对数值进行判断。用异常处理。不合法的数值要出现“这个数值是非法的”提示,不再进行运算。 interface Shape { //获取面积函数。 double getArea(); } class NoValueException extends RuntimeException { NoValueException(String message) { super(message); } } class Rec implements Shape { private int length; private int width; Rec(int length,int width) { if(length<=0 || width<=0) throw new NoValueException("非法数值"); this.length = length; this.width = width; } public double getArea() { return length*width; } } class Circle implements Shape { private int radius; private static final double PI = 3.14; Circle(int radius) { this.radius = radius; } public double getAge() { return radius*radius*PI; } } ---------------------- 7. 补足compare函数内的代码,不许添加其他函数。 class Circle { private static double pi=3.14; private double radius; public Circle(double r) { radius=r; } public static double compare(Circle[] cir) { //程序代码 /* 其实就是获取数组中的最大值。 */ // double max = cir[0].radius;// 初始化的是圆数组中的任意一个圆对象的半径。 int maxIndex = 0;//初始化的是数组任意一个角标。 for(int x=1; x<cir.length; x++) { if(cir[x].radius>cir[maxIndex].radius)//if(person.age > person1.age) if(age>age1) { maxIndex = x; } } return cir[maxIndex].radius; } } class TC { public static void main(String[] args) { Circle cir[]=new Circle[3];//创建一个Circle类型的数组,数组有3个元素。Circle[] arr = new Circle[3]; //分别对数组每一个元素进行初始化。 cir[0]=new Circle(1.0); cir[1]=new Circle(2.0); cir[2]=new Circle(4.0); System.out.println("最大的半径值是:"+Circle.compare(cir)); } }
作者:8亩田
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接.
本文如对您有帮助,还请多帮 【推荐】 下此文。
如果喜欢我的文章,请关注我的公众号
如果有疑问,请下面留言