Java中異常處理之try和catch代碼塊的使用
Java try和catch的使用
盡管由Java運(yùn)行時(shí)系統(tǒng)提供的默認(rèn)異常處理程序?qū)τ谡{(diào)試是很有用的,但通常你希望自己處理異常。這樣做有兩個(gè)好處。第一,它允許你修正錯(cuò)誤。第二,它防止程序自動(dòng)終止。大多數(shù)用戶對(duì)于在程序終止運(yùn)行和在無(wú)論何時(shí)錯(cuò)誤發(fā)生都會(huì)打印堆棧軌跡感到很煩惱(至少可以這么說(shuō))。幸運(yùn)的是,這很容易避免。
為防止和處理一個(gè)運(yùn)行時(shí)錯(cuò)誤,只需要把你所要監(jiān)控的代碼放進(jìn)一個(gè)try塊就可以了。緊跟著try塊的,包括一個(gè)說(shuō)明你希望捕獲的錯(cuò)誤類型的catch子句。完成這個(gè)任務(wù)很簡(jiǎn)單,下面的程序包含一個(gè)處理因?yàn)楸涣愠a(chǎn)生的ArithmeticException 異常的try塊和一個(gè)catch子句。
class Exc2 {
public static void main(String args[]) {
int d, a;
try { // monitor a block of code.
d = 0;
a = 42 / d;
System.out.println("This will not be printed.");
} catch (ArithmeticException e) { // catch divide-by-zero error
System.out.println("Division by zero.");
}
System.out.println("After catch statement.");
}
}
該程序輸出如下:
Division by zero. After catch statement.
注意在try塊中的對(duì)println( )的調(diào)用是永遠(yuǎn)不會(huì)執(zhí)行的。一旦異常被引發(fā),程序控制由try塊轉(zhuǎn)到catch塊。執(zhí)行永遠(yuǎn)不會(huì)從catch塊“返回”到try塊。因此,“This will not be printed?!?/p>
將不會(huì)被顯示。一旦執(zhí)行了catch語(yǔ)句,程序控制從整個(gè)try/catch機(jī)制的下面一行繼續(xù)。
一個(gè)try和它的catch語(yǔ)句形成了一個(gè)單元。catch子句的范圍限制于try語(yǔ)句前面所定義的語(yǔ)句。一個(gè)catch語(yǔ)句不能捕獲另一個(gè)try聲明所引發(fā)的異常(除非是嵌套的try語(yǔ)句情況)。
被try保護(hù)的語(yǔ)句聲明必須在一個(gè)大括號(hào)之內(nèi)(也就是說(shuō),它們必須在一個(gè)塊中)。你不能單獨(dú)使用try。
構(gòu)造catch子句的目的是解決異常情況并且像錯(cuò)誤沒(méi)有發(fā)生一樣繼續(xù)運(yùn)行。例如,下面的程序中,每一個(gè)for循環(huán)的反復(fù)得到兩個(gè)隨機(jī)整數(shù)。這兩個(gè)整數(shù)分別被對(duì)方除,結(jié)果用來(lái)除12345。最后的結(jié)果存在a中。如果一個(gè)除法操作導(dǎo)致被零除錯(cuò)誤,它將被捕獲,a的值設(shè)為零,程序繼續(xù)運(yùn)行。
// Handle an exception and move on.
import java.util.Random;
class HandleError {
public static void main(String args[]) {
int a=0, b=0, c=0;
Random r = new Random();
for(int i=0; i<32000; i++) {
try {
b = r.nextInt();
c = r.nextInt();
a = 12345 / (b/c);
} catch (ArithmeticException e) {
System.out.println("Division by zero.");
a = 0; // set a to zero and continue
}
System.out.println("a: " + a);
}
}
}
顯示一個(gè)異常的描述
Throwable重載toString( )方法(由Object定義),所以它返回一個(gè)包含異常描述的字符串。你可以通過(guò)在println( )中傳給異常一個(gè)參數(shù)來(lái)顯示該異常的描述。例如,前面程序的catch塊可以被重寫成
catch (ArithmeticException e) {
System.out.println("Exception: " + e);
a = 0; // set a to zero and continue
}
當(dāng)這個(gè)版本代替原程序中的版本,程序在標(biāo)準(zhǔn)javaJDK解釋器下運(yùn)行,每一個(gè)被零除錯(cuò)誤顯示下面的消息:
Exception: java.lang.ArithmeticException: / by zero
盡管在上下文中沒(méi)有特殊的值,顯示一個(gè)異常描述的能力在其他情況下是很有價(jià)值的——特別是當(dāng)你對(duì)異常進(jìn)行實(shí)驗(yàn)和調(diào)試時(shí)。
Java 多重catch語(yǔ)句的使用
某些情況,由單個(gè)代碼段可能引起多個(gè)異常。處理這種情況,你可以定義兩個(gè)或更多的catch子句,每個(gè)子句捕獲一種類型的異常。當(dāng)異常被引發(fā)時(shí),每一個(gè)catch子句被依次檢查,第一個(gè)匹配異常類型的子句執(zhí)行。當(dāng)一個(gè)catch語(yǔ)句執(zhí)行以后,其他的子句被旁路,執(zhí)行從try/catch塊以后的代碼開(kāi)始繼續(xù)。下面的例子設(shè)計(jì)了兩種不同的異常類型:
// Demonstrate multiple catch statements.
class MultiCatch {
public static void main(String args[]) {
try {
int a = args.length;
System.out.println("a = " + a);
int b = 42 / a;
int c[] = { 1 };
c[42] = 99;
} catch(ArithmeticException e) {
System.out.println("Divide by 0: " + e);
} catch(ArrayIndexOutOfBoundsException e) {
System.out.println("Array index oob: " + e);
}
System.out.println("After try/catch blocks.");
}
}
該程序在沒(méi)有命令行參數(shù)的起始條件下運(yùn)行導(dǎo)致被零除異常,因?yàn)閍為0。如果你提供一個(gè)命令行參數(shù),它將幸免于難,把a(bǔ)設(shè)成大于零的數(shù)值。但是它將導(dǎo)致ArrayIndexOutOf BoundsException異常,因?yàn)檎蛿?shù)組c的長(zhǎng)度為1,而程序試圖給c[42]賦值。
下面是運(yùn)行在兩種不同情況下程序的輸出:
C:\>java MultiCatch a = 0 Divide by 0: java.lang.ArithmeticException: / by zero After try/catch blocks. C:\>java MultiCatch TestArg a = 1 Array index oob: java.lang.ArrayIndexOutOfBoundsException After try/catch blocks.
當(dāng)你用多catch語(yǔ)句時(shí),記住異常子類必須在它們?nèi)魏胃割愔笆褂檬呛苤匾?。這是因?yàn)檫\(yùn)用父類的catch語(yǔ)句將捕獲該類型及其所有子類類型的異常。這樣,如果子類在父類后面,子類將永遠(yuǎn)不會(huì)到達(dá)。而且,Java中不能到達(dá)的代碼是一個(gè)錯(cuò)誤。例如,考慮下面的程序:
/* This program contains an error.
A subclass must come before its superclass in a series of catch statements. If not,unreachable code will be created and acompile-time error will result.
*/
class SuperSubCatch {
public static void main(String args[]) {
try {
int a = 0;
int b = 42 / a;
} catch(Exception e) {
System.out.println("Generic Exception catch.");
}
/* This catch is never reached because
ArithmeticException is a subclass of Exception. */
catch(ArithmeticException e) { // ERROR - unreachable
System.out.println("This is never reached.");
}
}
}
如果你試著編譯該程序,你會(huì)收到一個(gè)錯(cuò)誤消息,該錯(cuò)誤消息說(shuō)明第二個(gè)catch語(yǔ)句不會(huì)到達(dá),因?yàn)樵摦惓R呀?jīng)被捕獲。因?yàn)锳rithmeticException 是Exception的子類,第一個(gè)catch語(yǔ)句將處理所有的面向Exception的錯(cuò)誤,包括ArithmeticException。這意味著第二個(gè)catch語(yǔ)句永遠(yuǎn)不會(huì)執(zhí)行。為修改程序,顛倒兩個(gè)catch語(yǔ)句的次序。
相關(guān)文章
Mybatis實(shí)現(xiàn)一對(duì)多映射處理
MyBatis是一種流行的Java持久化框架,這篇文章主要為大家介紹了Mybatis如何實(shí)現(xiàn)一對(duì)多映射處理,文中的示例代碼講解詳細(xì),需要的可以參考下2023-08-08
SpringBoot 使用 OpenAPI3 規(guī)范整合 knife4j的詳細(xì)過(guò)程
Swagger工具集使用OpenAPI規(guī)范,可以生成、展示和測(cè)試基于OpenAPI規(guī)范的API文檔,并提供了生成客戶端代碼的功能,本文給大家介紹SpringBoot使用OpenAPI3規(guī)范整合knife4j的詳細(xì)過(guò)程,感興趣的朋友跟隨小編一起看看吧2023-12-12
Java數(shù)據(jù)導(dǎo)入功能之讀取Excel文件實(shí)例
這篇文章主要介紹了Java數(shù)據(jù)導(dǎo)入功能之讀取Excel文件實(shí)例,本文給出了jar包的下載地址以及讀取Excel文件的代碼實(shí)例,需要的朋友可以參考下2015-06-06
Netty分布式pipeline管道Handler的刪除邏輯操作
這篇文章主要為大家介紹了Netty分布式pipeline管道Handler的刪除邏輯操作,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03
java使用Base64實(shí)現(xiàn)文件加密解密
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)Base64給文件加密、解密,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-03-03
eclipse連接數(shù)據(jù)庫(kù)并實(shí)現(xiàn)用戶注冊(cè)登錄功能
這篇文章主要介紹了eclipse連接數(shù)據(jù)庫(kù)并實(shí)現(xiàn)用戶注冊(cè)登錄功能的相關(guān)資料,需要的朋友可以參考下2021-01-01

