Java中的異常棧軌跡和異常鏈


Java中允許對異常進行再次拋出,以提交給上一層進行處理,最為明顯的例子為Java的常規異常。

常規異常:有Java所定義的異常,不需要異常聲明,在未被try-catch的情況下,會被默認上報到main()方法。

Example:

public class TestException {
	TestException(int i) throws ExceptionA,ExceptionB{
		int a[]={0,}; a[1]=1;
	}
}

當從main()方法中調用TestException類的構造函數的時候,會得到以下的異常提示,從中可以看出異常的冒泡機制(這是一個棧信息)。

異常的冒泡上傳機制:當一個異常對象產生了以后,其會按照調用層次(一般是方法的調用層次)進行冒泡,直到被try-catch處理,或上報至main()方法,有編譯器進行提示。

Example:

firstThrow()提示錯誤的原因為, firstThrow()的函數聲明中包括了MyException的異常聲明,而secondThrow()中調用了firstThrow()卻未對可能拋出的異常對象提供任何處理方案,這是編譯器所不能允許發生的事情。(這也是為了保證底層異常對象在冒泡過程中,能得到合適的處理和重視!)

注意此截圖中,MyEclipse提供了兩種方式幫助fix這個程序這兩種方式常用的異常應對手段

1、本方法不給予處理,將異常上報給上一層。

 1 public class TestExceptionChain {
 2     void firstThrow() throws MyException
 3     {
 4         print("Oringinally creat a MyException and throw it out");
 5         throw new MyException();
 6         
 7     }
 8     void secondThrow() throws MyException
 9     {
10         firstThrow();
11     }
12     TestExceptionChain() throws MyException{
13         secondThrow();
14     }
15 }
 1 public static void main(String[] args) {
 2         try{
 3          
 4             TestExceptionChain testExceptionChain=new TestExceptionChain();
 5         }
 6         catch(MyException e)
 7         {
 8             e.printStackTrace();
 9             print("Catch a my exception!");
10         }
11 
12     }

控制台的輸出為:

從異常棧的記錄信息可以發現,與代碼相對應的異常拋出機制和次序:

firstThrow()產生MyException對象->異常冒泡至調用其的secondThrow()->異常冒泡至調用secondThrow()的TestExceptionChain的構造方法->冒泡至printtry的main()方法。

注意到:異常對象一直被拋出,直至在printtry的mian()方法中被try-catch捕獲!

 

2、try-catch方式,捕捉上報的異常,而后進行相應處理或拋出另一異常。

2、1捕獲異常后,進行相應處理。

Example:

 1 public class TestExceptionChain {
 2     void firstThrow() throws MyException
 3     {
 4         print("Oringinally creat a MyException and throw it out");
 5         throw new MyException();
 6     }
 7     void secondThrow()
 8     {
 9         try
10         {
11         firstThrow();
12         }
13         catch (MyException e)
14         {
15             print("I have just caught a MyExcepton,but i want to do nothing for it");
16             e.printStackTrace();
17         }
18     }
19     TestExceptionChain(){
20         secondThrow();
21     }

從圖中可以發現,異常在secondThrow() 中被try-catch模塊捕獲,並執行了相應的處理操作,所以其函數聲明中無需添加異常聲明,異常不會被上報

故mian()方法被改寫成了以下的代碼:

1 TestExceptionChain testExceptionChain=new TestExceptionChain();

注意此處異常棧的信息,表示的是異常產生的層次信息,並非異常信息的上報層次,因為其已經在secondThorow()中被捕獲處理。

 

2.2 捕獲異常后,拋出另一個異常。

Example:

 1 public class TestExceptionChain {
 2     void firstThrow() throws MyException
 3     {
 4         print("Oringinally creat a MyException and throw it out");
 5         throw new MyException();
 6     }
 7     void secondThrow() throws YouException
 8     {
 9         try
10         {
11         firstThrow();
12         }
13         catch (MyException e)
14         {
15             print("I have just caught a MyExcepton,but i want to create a YouException and throw it out");
16             e.printStackTrace();
17             throw new YouException();
18         }
19     }
20     TestExceptionChain() throws YouException{
21         secondThrow();
22     }
23 }
24 class MyException extends Exception{}
25 class YouException extends Exception{}

從異常棧信息中可以發現,新拋出的YouException對象是從secondThrow()中開始的。

 

*Java中還提供了fillInStackTrace()方法,用於對捕獲的異常的棧信息進行重寫。

Example:

 1 try{
 2          
 3             TestExceptionChain testExceptionChain=new TestExceptionChain();
 4         }
 5         catch(YouException e)
 6         {
 7             print("Catch a YouException!");
 8             e.fillInStackTrace();
 9             e.printStackTrace();
10             
11         }

 由於使用了fillInstack()方法,關於YouException的異常信息被重寫,從其被從寫處重新記錄!

 

 

 

 

 

 

 

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM