国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Java > 正文

Java進(jìn)階教程之異常處理

2019-11-26 15:26:19
字體:
供稿:網(wǎng)友

程序很難做到完美,不免有各種各樣的異常。比如程序本身有bug,比如程序打印時打印機沒有紙了,比如內(nèi)存不足。為了解決這些異常,我們需要知道異常發(fā)生的原因。對于一些常見的異常,我們還可以提供一定的應(yīng)對預(yù)案。C語言中的異常處理是簡單的通過函數(shù)返回值來實現(xiàn)的,但返回值代表的含義往往是由慣例決定的。程序員需要查詢大量的資料,才可能找到一個模糊的原因。面向?qū)ο笳Z言,比如C++, Java, Python往往有更加復(fù)雜的異常處理機制。這里討論Java中的異常處理機制。

Java異常處理

異常處理

Java的異常處理機制很大一部分來自C++。它允許程序員跳過暫時無法處理的問題,以繼續(xù)后續(xù)的開發(fā),或者讓程序根據(jù)異常做出更加聰明的處理。

Java使用一些特殊的對象來代表異常狀況,這樣對象稱為異常對象。當(dāng)異常狀況發(fā)生時,Java會根據(jù)預(yù)先的設(shè)定,拋出(throw)代表當(dāng)前狀況的對象。所謂的拋出是一種特殊的返回方式。該線程會暫停,逐層退出方法調(diào)用,直到遇到異常處理器(Exception Handler)。異常處理器可以捕捉(catch)的異常對象,并根據(jù)對象來決定下一步的行動,比如:

提醒用戶
處理異常
繼續(xù)程序
退出程序
......

異常處理器看起來如下,它由try, catch, finally以及隨后的程序塊組成。finally不是必須的。

try { ...;}catch() { ...;}catch() { ...;}finally { ...;}

這個異常處理器監(jiān)視try后面的程序塊。catch的括號有一個參數(shù),代表所要捕捉的異常的類型。catch會捕捉相應(yīng)的類型及其衍生類。try后面的程序塊包含了針對該異常類型所要進(jìn)行的操作。try所監(jiān)視的程序塊可能拋出不止一種類型的異常,所以一個異常處理器可以有多個catch模塊。finally后面的程序塊是無論是否發(fā)生異常,都要執(zhí)行的程序。

我們在try中放入可能出錯,需要監(jiān)視的程序,在catch中設(shè)計應(yīng)對異常的方案。

 下面是一段使用到異常處理的部分Java程序。try部分的程序是從一個文件中讀取文本行。在讀取文件的過程中,可能會有IOException發(fā)生:

BufferedReader br = new BufferedReader(new FileReader("file.txt"));try {  StringBuilder sb = new StringBuilder();  String line = br.readLine();  while (line != null) {    sb.append(line);    sb.append("/n");    line = br.readLine();  }  String everything = sb.toString();} catch(IOException e) {  e.printStackTrace();  System.out.println("IO problem");}finally {  br.close();}

如果我們捕捉到IOException類對象e的時,可以對該對象操作。比如調(diào)用對象的printStackTrace(),打印當(dāng)前棧的狀況。此外,我們還向中端打印了提示"IO problem"。

無論是否有異常,程序最終會進(jìn)入finally塊中。我們在finally塊中關(guān)閉文件,清空文件描述符所占據(jù)的資源。

 異常的類型

Java中的異常類都繼承自Trowable類。一個Throwable類的對象都可以拋出(throw)。

 

橙色: unchecked; 藍(lán)色: checked

Throwable對象可以分為兩組。一組是unchecked異常,異常處理機制往往不用于這組異常,包括:

1.Error類通常是指Java的內(nèi)部錯誤以及如資源耗盡的錯誤。當(dāng)Error(及其衍生類)發(fā)生時,我們不能在編程層面上解決Error,所以應(yīng)該直接退出程序。

2.Exception類有特殊的一個衍生類RuntimeException。RuntimeException(及其衍生類)是Java程序自身造成的,也就是說,由于程序員在編程時犯錯。RuntimeException完全可以通過修正Java程序避免。比如將一個類型的對象轉(zhuǎn)換成沒有繼承關(guān)系的另一個類型,即ClassCastException。這類異常應(yīng)該并且可以避免。

剩下的是checked異常。這些類是由編程與環(huán)境互動造成程序在運行時出錯。比如讀取文件時,由于文件本身有錯誤,發(fā)生IOException。再比如網(wǎng)絡(luò)服務(wù)器臨時更改URL指向,造成MalformedURLException。文件系統(tǒng)和網(wǎng)絡(luò)服務(wù)器是在Java環(huán)境之外的,并不是程序員所能控制的。如果程序員可以預(yù)期異常,可以利用異常處理機制來制定應(yīng)對預(yù)案。比如文件出問題時,提醒系統(tǒng)管理員。再比如在網(wǎng)絡(luò)服務(wù)器出現(xiàn)問題時,提醒用戶,并等待網(wǎng)絡(luò)服務(wù)器恢復(fù)。異常處理機制主要是用于處理這樣的異常。

拋出異常

在上面的程序中,異常來自于我們對Java IO API的調(diào)用。我們也可以在自己的程序中拋出異常,比如下面的battery類,有充電和使用方法:

 

public class Test{  public static void main(String[] args)  {    Battery aBattery = new Battery();    aBattery.chargeBattery(0.5);    aBattery.useBattery(-0.5);  }}class Battery {  /**   * increase battery   */  public void chargeBattery(double p)  {    // power <= 1    if (this.power + p < 1.) {      this.power = this.power + p;    }    else {      this.power = 1.;    }  }  /**   * consume battery   */  public boolean useBattery(double p)  {    try {      test(p);    }    catch(Exception e) {      System.out.println("catch Exception");      System.out.println(e.getMessage());      p = 0.0;    }    if (this.power >= p) {      this.power = this.power - p;      return true;    }    else {      this.power = 0.0;      return false;    }  }  /**   * test usage   */  private void test(double p) throws Exception // I just throw, don't handle  {    if (p < 0) {      Exception e = new Exception("p must be positive");      throw e;    }  }  private double power = 0.0; // percentage of battery}

useBattery()表示使用電池操作。useBattery()方法中有一個參數(shù),表示使用的電量。我們使用test()方法測試該參數(shù)。如果該參數(shù)為負(fù)數(shù),那么我們認(rèn)為有異常,并拋出。

在test中,當(dāng)有異常發(fā)生時(p < 0),我們創(chuàng)建一個Exception對象e,并用一個字符串作為參數(shù)。字符串中包含有異常相關(guān)的信息,該參數(shù)不是必需的。使用throw將該Exception對象拋出。

我們在useBattery()中有異常處理器。由于test()方法不直接處理它產(chǎn)生的異常,而是將該異常拋給上層的useBattery(),所以在test()的定義中,我們需要throws Exception來說明。

(假設(shè)異常處理器并不是位于useBattery()中,而是在更上層的main()方法中,我們也要在useBattery()的定義中增加throws Exception。)

 在catch中,我們使用getMessage()方法提取其異常中包含的信息。上述程序的運行結(jié)果如下:

catch Exceptionp must be positive

異常處理器中,我們會捕捉任意Exception類或者其衍生類異常。這往往不利于我們識別問題,特別是一段程序可能拋出多種異常時。我們可以提供一個更加具體的類來捕捉。

 自定義異常

我們可以通過繼承來創(chuàng)建新的異常類。在繼承時,我們往往需要重寫構(gòu)造方法。異常有兩個構(gòu)造方法,一個沒有參數(shù),一個有一個String參數(shù)。比如:

class BatteryUsageException extends Exception{  public BatteryUsageException() {}  public BatteryUsageException(String msg) {    super(msg);  }}

我們可以在衍生類中提供更多異常相關(guān)的方法和信息。

 在自定義異常時,要小心選擇所繼承的基類。一個更具體的類要包含更多的異常信息,比如IOException相對于Exception。

 總結(jié)

異常處理是在解決問題,同時也是在制造問題。大型項目中,過多、過細(xì)的異常處理往往會導(dǎo)致程序變得一團(tuán)糟。異常處理的設(shè)計并不簡單,并需要謹(jǐn)慎使用。

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 大新县| 山阳县| 达拉特旗| 永嘉县| 汉中市| 克东县| 德格县| 重庆市| 金阳县| 天门市| 惠东县| 山东省| 大英县| 辽阳市| 屏边| 鹤庆县| 崇信县| 大同市| 邵阳市| 积石山| 五寨县| 滨海县| 渝中区| 汉中市| 游戏| 辽宁省| 健康| 宾川县| 莱阳市| 遵义县| 昭平县| 长沙市| 嵊泗县| 涿鹿县| 延长县| 舟山市| 福州市| 象州县| 哈密市| 贵溪市| 潮安县|