java中的异常涉及到父子类的问题,可以归纳为一句话:子类的构造函数抛出的异常必须包含父类的异常,子类的方法可以选择抛出“范围小于等于”父类的异常或不抛出异常。
1. 为什么构造函数必须抛出包含父类的异常?
在《thingking in java》中有这么一段话:
异常限制:当覆盖方法时,只能抛出在基类方法的异常说明中列出的那些异常
异常限制对构造器不起作用,你会发现StormyInning的构造器可以抛出任何异常,而不必理会基类构造html" target="_blank">函数所抛出的异常。然而因为必须构造函数必须以这样或那样的方式被调用,子类构造函数的异常说明必须包含基类构造器的异常说明
这段话起初一开比较绕,但是嘻嘻看一遍就会明白:
首先,异常说明只针对覆盖方法,而构造函数明显不在这个范围,所以子类构造函数可以抛出任何异常,而不用顾及父类构造函数所抛出的异常。但是在new 一个子类对象的时候,父类构造函数一定会被调用,所以子类构造函数调用的对应的父类构造函数所抛出的异常就必须考虑在内,此时又因为“子类构造函数无法捕获父类构造函数所抛出的异常(后面会提)”,所以子类构造函数必须抛出这个异常。
class SomeException extends Exception{} class TheOtherException extends Exception{} class BaseC { public BaseC()throws SomeException{} public BaseC(int a)throws TheOtherException{} } class SubC extends BaseC { public SubC() throws SomeException //如果不抛出异常就会报错 { super(); //由于调用的基类的默认构造函数, 所以要抛出SomeException //super(37) ; //如果将super()替换成这里,就必须抛出TheOtherException } }
2. 为什么子类构造函数无法捕获父类构造函数所抛出的异常?
因为子类如果想要捕获父类抛出的异常,就必须显示地调用super() ; 或者super(xxx...); 然而super()和this()这些都有一个特性, 就是必须将他们放在第一行, 这与try{}catch{}相矛盾, 所以无法捕获
3. 当子类继承的父类和接口存在相同的方法名时,这时的处理方式就必须遵循异常限制。
class SomeException extends Exception{} class TheOtherException extends Exception{} interface InterF { public void function()throws TheOtherException; } class BaseC { public void function()throws SomeException{} } class SubC extends BaseC implements InterF { //此时只能选择不抛出异常 public void function(){} //报错:Exception SomeException is not compatible with throws clause in InterF.function() // public void function()throws SomeException{} ; //报错:Exception TheOtherException is not compatible with throws clause in BaseC.function() // public void function()throws TheOtherException{} ; }
4. 为什么子类只能抛出在基类方法的异常说明中列出的那些异常?
因为子类存在向上转化成父类的可能性,如果允许子类随意抛出异常的话,那么向上转化成父类时,该方法的接口(姑且这么叫吧)会变成父类的方法类型,此时问题来了,子类会抛出异常,而父类却无法对该异常做出处理,所以为保证对象的可替换型,强制要求“只能抛出在基类方法的异常说明中列出的那些异常”。
这里所说的“那些异常”还包括这些异常的子异常!
5. 这一点不知道算不算,也许是我比较愚钝吧,我在看的时候想了好久才明白过来,姑且记下来吧。
class SomeException extends Exception{} class BaseC { public void function()throws SomeException{}//如果这里抛出的异常是一个运行时异常子类就可以不对其进行异常处理 } class SubC extends BaseC {<BR>//这两个function()所进行的super.function()的都属于正常的函数调用,不属于异常处理的范围, 但是这个函数本身要符合异常处理的规范! /* public void function()throws SomeException { super.function(); } */ public void function() { try { super.function() ; } catch(SomeException e) { e.printStackTrace(); } } }
以上这篇基于java涉及父子类的异常详解就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持小牛知识库。
本文向大家介绍浅谈java异常处理(父子异常的处理),包括了浅谈java异常处理(父子异常的处理)的使用技巧和注意事项,需要的朋友参考一下 我当初学java异常处理的时候,对于父子异常的处理,我记得几句话“子类方法只能抛出父类方法所抛出的异常或者是其子异常,子类构造器必须要抛出父类构造器的异常或者其父异常”。那个时候还不知道子类方法为什么要这样子抛出异常,后来通过学习《Thinking in Ja
本文向大家介绍Java异常处理运行时异常(RuntimeException)详解及实例,包括了Java异常处理运行时异常(RuntimeException)详解及实例的使用技巧和注意事项,需要的朋友参考一下 Java异常处理运行时异常(RuntimeException)详解及实例 RuntimeException RunntimeException的子类: ClassCastException
我需要创建一个容器,为我存储泛型类型的元素提供一种方式,有点像这个有效的Java模式,但是存储泛型的东西 有没有可能创建一个类型安全异构容器,其中涉及到泛型类型的东西? 我可以将添加为方法参数。示例: 有可能做到这一点吗? 怎么做,或者为什么不做?它是Java不做的事情还是基本的事情(所以没有语言可以做,或者只是做没有意义)
本文向大家介绍Java 常用类解析:java异常机制,异常栈,异常处理方式,异常链,异常丢失详解,包括了Java 常用类解析:java异常机制,异常栈,异常处理方式,异常链,异常丢失详解的使用技巧和注意事项,需要的朋友参考一下 1、java标准异常概述 Throwable表示任何可以作为异常被抛出的类,有两个子类Error和Exception。从这两个类的源代码中可以看出,这两个类并没有添加新的方
本文向大家介绍Java自定义异常类的实例详解,包括了Java自定义异常类的实例详解的使用技巧和注意事项,需要的朋友参考一下 Java自定义异常类的实例详解 为什么要自己编写异常类?假如jdk里面没有提供的异常,我们就要自己写。我们常用的类ArithmeticException,NullPointerException,NegativeArraySizeException,ArrayIndexout
Java异常简介 Java异常是Java提供的一种识别及响应错误的一致性机制。 Java异常机制可以使程序中异常处理代码和正常业务代码分离,保证程序代码更加优雅,并提高程序健壮性。在有效使用异常的情况下,异常能清晰的回答what, where, why这3个问题:异常类型回答了“什么”被抛出,异常堆栈跟踪回答了“在哪“抛出,异常信息回答了“为什么“会抛出。 Java异常机制用到的几个关键字:try