我當初學java異常處理的時候,對於父子異常的處理,我記得幾句話“子類方法只能拋出父類方法所拋出的異常或者是其子異常,子類構造器必須要拋出父類構造器的異常或者其父異常”。那個時候還不知道子類方法為什麼要這樣子拋出異常,後來通過學習《Thinking in Java》,我才明白其中的道理,現在我再來溫習一下。
一、子類方法只能拋出父類方法的異常或者是其子異常
對於這種限制,主要是因為子類在做向上轉型的時候,不能正確地捕獲異常
package thinkinginjava;
public abstract class InningDemo1 {
public void walk() throws BaseException{}
public abstract void sing() throws BaseException;
}
class BaseException extends Exception{}
class SubException1 extends BaseException{}
class CupException extends Exception{}
package thinkinginjava;
public interface OtherFunction {
public void task() throws RuntimeException;
}
package thinkinginjava;
public class SubInningDemo1 extends InningDemo1 implements OtherFunction{
//子類方法可以拋出父類方法的異常
@Override
public void walk() throws BaseException{}
//但不能拋出父類中沒有的異常,否則編譯會出錯
//public void walk() throws Exception{}
//子類方法可以拋出父類方法的子異常
@Override
public void sing() throws SubException1{}
//當實現的接口和父類中的方法都有異常時,可以選擇不拋出異常
public void task(){}
}
就拿這個例子來說,假如子類中有這方法 public void walk() throws CupException{},拋出了父類方法沒有的異常,我們使用父類的引用指向子類
public void f(){
InningDemo1 inn = new SubInningDemo1();
inn.walk() ;//父類調用walk()方法時,並不知道它會拋出CupException,從而f()方法不知如何捕獲異常。因此在編譯期就要防止子類方法亂拋異常。
}
從上面的例子中,我們也可以看到,子類方法可以不拋出異常
二、子類構造器必須要拋出父類構造器的異常或者其父異常
這是因為子類構造器中默認添加了父類的構造器
package thinkinginjava;
public abstract class InningDemo2 {
public InningDemo2() throws SubException{
}
}
class FatherException extends Exception{}
class SubException extends FatherException{}
class PenException extends Exception{}
package thinkinginjava;
public class SubInningDemo2 extends InningDemo2{
public SubInningDemo2() throws FatherException {
//子類構造器中默認添加了父類的構造器,所以需要拋出父類的異常或者其父異常
//super();
}
}
三、異常丟失
1、在finally中拋出異常,可能會之前拋出的異常丟失
package thinkinginjava;
public class FinallyException {
public static void main(String[] args){
try{
try{
throw new RedException();
}finally{
//把上一個異常覆蓋掉
throw new BlueException();
}
}catch(Exception e){
System.out.println(e);
}
}
}
class RedException extends Exception{}
class BlueException extends Exception{}
運行結果:thinkinginjava.BlueException
2、在finally中使用return,不會拋出異常
package thinkinginjava;
public class ReturnException {
public static void main(String[] args){
try{
throw new Exception();
}finally{
return;
}
}
}
以上代碼我們看到它拋出了異常,但運行時不會有任何輸出