Java catch與throw同時使用的操作
一、應用背景
在實際應用中,處理異常往往需要更加復雜的處理——當一個異常出現時,單靠某個方法無法完全處理該異常,必須由幾個方法協作才能完全處理該異常,也就是說,在異常出現的當前方法中,程序隻能對異常進行部分處理,還有些處理需要在方法的調用者中才能完成,所以應該再次拋出異常,讓該方法的調用者也能捕獲到異常。
為瞭實現這種通過多個方法協作處理同一異常的情形,可以catch塊中結合throw語句來完成。
二、應用舉例
1 代碼示例
AuctionTest.java
public class AuctionTest { private double initPrice = 30.0; // 因為該方法中顯式拋出瞭AuctionException異常, // 所以此處需要聲明拋出AuctionException異常 public void bid(String bidPrice) throws AuctionException { double d = 0.0; try { d = Double.parseDouble(bidPrice); } catch (Exception e) { // 此處完成本方法中可以對異常執行的修復處理, // 此處僅僅是在控制臺打印異常跟蹤棧信息。 e.printStackTrace(); // 再次拋出自定義異常 throw new AuctionException("競拍價必須是數值," + "不能包含其他字符!"); } if (initPrice > d) { throw new AuctionException("競拍價比起拍價低," + "不允許競拍!"); } initPrice = d; } public static void main(String[] args) { AuctionTest at = new AuctionTest(); try { at.bid("df"); } catch (AuctionException ae) { // 再次捕捉到bid方法中的異常。並對該異常進行處理 System.err.println(ae.getMessage()); } } }
AuctionException.java
public class AuctionException extends Exception { // 無參數的構造器 public AuctionException(){} //① // 帶一個字符串參數的構造器 public AuctionException(String msg) //② { super(msg); } }
2 運行結果
java.lang.NumberFormatException: For input string: "df" at sun.misc.FloatingDecimal.readJavaFormatString(FloatingDecimal.java:1224) at java.lang.Double.parseDouble(Double.java:510) at AuctionTest.bid(AuctionTest.java:16) at AuctionTest.main(AuctionTest.java:39)
競拍價必須是數值,不能包含其他字符!
3 結果說明
上面程序bid對應catch塊捕獲到異常後,系統打印瞭該異常的跟蹤棧信息,接著拋出一個AuctionException異常,通知該方法調用者再次處理AuctionException異常。
所有程序中的main方法,也就是bid方法調用者再次捕獲AuctionException異常並將該異常詳細描述信息輸出到標準錯誤輸出。
補充:Java try-catch、throw和throws的幾點想法
以前寫代碼,很少用到異常,後來發現這種習慣是錯的。異常也是一種信息,並不是錯誤。
1:先寫個簡單的類:
package com.exception.demo; public class Main { public static void main(String[] args) { Main main = new Main(); } public void methodTry() { } public void methodThrow() { } }
初始環境就是這麼簡答。
2:下面給方法methodTry加上方法主體:
public static void main(String[] args) { Main main = new Main(); main.methodTry(); } public void methodTry() { int a=10; int b=0; int c=a/b; }
剛開始學代碼的時候都會寫這個方法,會拋出一個異常:
控制臺很清楚的告訴我們,被除數不能為0.但是如果我們想自己獲取這個異常,然後做些操作呢?比如說 如果這個方法體有問題,我就做一個輸出。
public void methodTry() { try { int a=10; int b=0; int c=a/b; System.out.println(c); } catch (Exception e) { System.out.println("這個方法體有問題:"+e.getMessage()); } }
這個時候就用到瞭try-catch,手動的捕獲這個異常,然後進行我們需要的操作。畢竟異常分很多種,並不是所有的異常都是我們不需要的。
比如說對用戶登錄來說,登錄成功 登錄失敗兩種結果,登錄失敗又分為重復登錄,賬號密碼不匹配等。
我們可以把這些失敗全都寫成Exception。當成功的時候就直接返回,失敗的時候拋出異常,這個可比我們寫好多返回值簡單多瞭。
接著說try-catch
我們手動的捕獲瞭這個異常。上面的代碼告訴我們,當try-catch塊中有異常時,異常後面的代碼是不會執行的。try-catch還有什麼好處?回滾。
3:throw
public static void main(String[] args) { Main main = new Main(); main.methodThrow(); } public void methodThrow() { throw new Exception("這裡有異常"); }
其實當我簡單的拋出個異常的時候,throw new Exception()這裡會報錯,現在看一個報錯信息:
顯示讓我們選擇是throws 還是try-catch。
我們寫好的throw是什麼意思呢? 其實跟a/0是一個意思,都是拋出一個異常,隻不過一個是jdk已經定義好的異常,被除數不能為0.一個是我們手動拋出的異常。
先用try-catch試試看
public void methodThrow() { try { throw new Exception("這裡有異常"); } catch (Exception e) { System.out.println("MethodThrow:"+e.getMessage()); } }
重點在於手動拋出異常後,我們要在catch中進行處理,在catch中寫我們的部門邏輯代碼。
4:throws
剛才我們選擇的是try-catch,現在選擇throws
public static void main(String[] args) throws Exception { Main main = new Main(); main.methodThrow(); } public void methodThrow() throws Exception { throw new Exception("這裡有異常"); }
方法methodThrow throws Exception之後,他的父類就面臨著兩種情況,要麼try-catch 要麼throws這個異常。這種情況跟methodThrow中手動拋出異常遇到的問題是一樣的。
看來可以這麼理解:
throw是手動拋出異常,跟 被除數不能為0 數組下標越界等異常一樣,都是異常。
try-catch是在catch中手動捕獲異常,然後進行一些操作。比如說輸出異常信息,打印錯誤日志等。
throws是往上級拋出異常,我的方法methodThrow有異常,但是在這個方法中我不進行處理瞭,讓上級進行處理吧。然後就跑到main函數那去瞭。
對main函數來說,可以throws讓系統進行處理,也可以自己處理這個異常啊。
main.methodThrow()和a/0 、throw new Execption()沒什麼區別,都是有異常。
其實可以整體的寫:
public static void main(String[] args){ Main main = new Main(); try { main.methodThrow(); } catch (Exception e) { System.out.println(e.getMessage()); } } public void methodThrow() throws Exception { throw new Exception("這裡有異常"); }
方法中的異常,到main函數中再進行處理。
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。