基于java涉及父子類的異常詳解
java中的異常涉及到父子類的問題,可以歸納為一句話:子類的構(gòu)造函數(shù)拋出的異常必須包含父類的異常,子類的方法可以選擇拋出“范圍小于等于”父類的異?;虿粧伋霎惓?。
1. 為什么構(gòu)造函數(shù)必須拋出包含父類的異常?
在《thingking in java》中有這么一段話:
異常限制:當(dāng)覆蓋方法時,只能拋出在基類方法的異常說明中列出的那些異常
異常限制對構(gòu)造器不起作用,你會發(fā)現(xiàn)StormyInning的構(gòu)造器可以拋出任何異常,而不必理會基類構(gòu)造函數(shù)所拋出的異常。然而因為必須構(gòu)造函數(shù)必須以這樣或那樣的方式被調(diào)用,子類構(gòu)造函數(shù)的異常說明必須包含基類構(gòu)造器的異常說明
這段話起初一開比較繞,但是嘻嘻看一遍就會明白:
首先,異常說明只針對覆蓋方法,而構(gòu)造函數(shù)明顯不在這個范圍,所以子類構(gòu)造函數(shù)可以拋出任何異常,而不用顧及父類構(gòu)造函數(shù)所拋出的異常。但是在new 一個子類對象的時候,父類構(gòu)造函數(shù)一定會被調(diào)用,所以子類構(gòu)造函數(shù)調(diào)用的對應(yīng)的父類構(gòu)造函數(shù)所拋出的異常就必須考慮在內(nèi),此時又因為“子類構(gòu)造函數(shù)無法捕獲父類構(gòu)造函數(shù)所拋出的異常(后面會提)”,所以子類構(gòu)造函數(shù)必須拋出這個異常。
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(); //由于調(diào)用的基類的默認(rèn)構(gòu)造函數(shù), 所以要拋出SomeException //super(37) ; //如果將super()替換成這里,就必須拋出TheOtherException } }
2. 為什么子類構(gòu)造函數(shù)無法捕獲父類構(gòu)造函數(shù)所拋出的異常?
因為子類如果想要捕獲父類拋出的異常,就必須顯示地調(diào)用super() ; 或者super(xxx...); 然而super()和this()這些都有一個特性, 就是必須將他們放在第一行, 這與try{}catch{}相矛盾, 所以無法捕獲
3. 當(dāng)子類繼承的父類和接口存在相同的方法名時,這時的處理方式就必須遵循異常限制。
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. 為什么子類只能拋出在基類方法的異常說明中列出的那些異常?
因為子類存在向上轉(zhuǎn)化成父類的可能性,如果允許子類隨意拋出異常的話,那么向上轉(zhuǎn)化成父類時,該方法的接口(姑且這么叫吧)會變成父類的方法類型,此時問題來了,子類會拋出異常,而父類卻無法對該異常做出處理,所以為保證對象的可替換型,強制要求“只能拋出在基類方法的異常說明中列出的那些異?!?。
這里所說的“那些異常”還包括這些異常的子異常!
5. 這一點不知道算不算,也許是我比較愚鈍吧,我在看的時候想了好久才明白過來,姑且記下來吧。
class SomeException extends Exception{} class BaseC { public void function()throws SomeException{}//如果這里拋出的異常是一個運行時異常子類就可以不對其進行異常處理 } class SubC extends BaseC {<BR>//這兩個function()所進行的super.function()的都屬于正常的函數(shù)調(diào)用,不屬于異常處理的范圍, 但是這個函數(shù)本身要符合異常處理的規(guī)范! /* public void function()throws SomeException { super.function(); } */ public void function() { try { super.function() ; } catch(SomeException e) { e.printStackTrace(); } } }
以上這篇基于java涉及父子類的異常詳解就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java 梳理總結(jié)關(guān)于static關(guān)鍵字常見問題
static關(guān)鍵字基本概念我們可以一句話來概括:方便在沒有創(chuàng)建對象的情況下來進行調(diào)用。也就是說:被static關(guān)鍵字修飾的不需要創(chuàng)建對象去調(diào)用,直接根據(jù)類名就可以去訪問,讓我們來了解一下你可能還不知道情況2022-04-04SpringMVC基于阻塞隊列LinkedBlockingQueue的同步長輪詢功能實現(xiàn)詳解
這篇文章主要介紹了SpringMVC基于阻塞隊列LinkedBlockingQueue的同步長輪詢功能實現(xiàn)詳解,本文介紹的也是生產(chǎn)者消費者的一種實現(xiàn),生產(chǎn)者不必是一個始終在執(zhí)行的線程,它可以是一個接口,接受客戶端的請求,向隊列中插入消息,需要的朋友可以參考下2023-07-07MyBatis入門學(xué)習(xí)教程(一)-MyBatis快速入門
MyBatis是一個支持普通SQL查詢,存儲過程和高級映射的優(yōu)秀持久層框架,這篇文章主要給大家分享MyBatis入門學(xué)習(xí)教程(一)-MyBatis快速入門,需要的朋友可以參考下2015-08-08Spring Security 將用戶數(shù)據(jù)存入數(shù)據(jù)庫
這篇文章主要介紹了Spring Security 如何將用戶數(shù)據(jù)存入數(shù)據(jù)庫,幫助大家更好的理解和學(xué)習(xí)Spring Security,感興趣的朋友可以了解下2020-09-09