欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

通過實(shí)踐了解如何處理Java異常

 更新時(shí)間:2019年05月23日 16:16:48   作者:銀河1號  
Java中的異常處理不是一個(gè)簡單的主題。初學(xué)者發(fā)現(xiàn)它很難理解,甚至有經(jīng)驗(yàn)的開發(fā)者也可以花幾個(gè)小時(shí)討論如何以及應(yīng)該拋出或處理哪些異常。下面我們通過實(shí)踐來了解如何解決異常

大多數(shù)團(tuán)隊(duì)都使用了幾種最佳實(shí)踐。以下是幫助你入門或改進(jìn)異常處理的9個(gè)最重要的內(nèi)容。

1.在finally塊中清理資源或使用Try-With-Resource語句

在try塊中使用資源是很頻繁的,比如InputStream,之后需要關(guān)閉它。這些情況中的一個(gè)常見錯(cuò)誤是在try塊結(jié)束時(shí)關(guān)閉資源。

public void doNotCloseResourceInTry() {
 FileInputStream inputStream = null;
 try {
 File file = new File("./tmp.txt");
 inputStream = new FileInputStream(file);
 // use the inputStream to read a file
 // do NOT do this
 inputStream.close();
 } catch (FileNotFoundException e) {
 log.error(e);
 } catch (IOException e) {
 log.error(e);
 }
}

問題是只要沒有拋出異常,這種方法似乎完全正常。try塊中的所有語句都將被執(zhí)行,資源將被關(guān)閉。

但是你添加了try塊是有原因的。你調(diào)用一個(gè)或多個(gè)可能拋出異常的方法,或者你自己拋出異常。這意味著你可能無法到達(dá)try塊的末尾。因此,你將不會關(guān)閉資源。

因此,你應(yīng)該將所有清理代碼放入finally塊或使用try-with-resource語句。

使用Finally塊

與try塊的最后幾行相比,finally塊始終執(zhí)行。這可以在成功執(zhí)行try塊之后或在catch塊中處理異常之后發(fā)生。因此,你可以確保清理所有已打開的資源。

public void closeResourceInFinally() {
 FileInputStream inputStream = null;
 try {
 File file = new File("./tmp.txt");
 inputStream = new FileInputStream(file);
 // use the inputStream to read a file
 } catch (FileNotFoundException e) {
 log.error(e);
 } finally {
 if (inputStream != null) {
 try {
 inputStream.close();
 } catch (IOException e) {
 log.error(e);
 }
 }
 }
}

Java 7的Try-With-Resource

另一種選擇是try-with-resource語句,我在Java異常處理的介紹中對此進(jìn)行了更詳細(xì)的解釋。

如果資源實(shí)現(xiàn)AutoCloseable接口,則可以使用它。這就是大多數(shù)Java標(biāo)準(zhǔn)資源所做的事情。當(dāng)你在try子句中打開資源時(shí),它將在try塊執(zhí)行后自動關(guān)閉,或者處理異常。

public void automaticallyCloseResource() {
 File file = new File("./tmp.txt");
 try (FileInputStream inputStream = new FileInputStream(file);) {
 // use the inputStream to read a file
 } catch (FileNotFoundException e) {
 log.error(e);
 } catch (IOException e) {
 log.error(e);
 }
}

2.特定異常

拋出的異常越具體越好。請記住,不明白你代碼的同事,或者你可能在幾個(gè)月后需要調(diào)用你的方法并處理異常。

因此,請務(wù)必提供盡可能多的信息。這使你的API更易于理解。因此,你的方法的調(diào)用者將能夠更好地處理異?;蛲ㄟ^額外的檢查來避免它。

因此,總是嘗試找到最適合你的異常事件的類,例如拋出NumberFormatException而不是IllegalArgumentException。并避免拋出非特定的異常。

public void doNotDoThis() throws Exception {
 ...
}
public void doThis() throws NumberFormatException {
 ...
}

3.記錄你聲明的異常

無論何時(shí)在方法簽名中指定異常,都應(yīng)該在Javadoc中記錄它。這與以前的最佳實(shí)踐具有相同的目標(biāo):為調(diào)用者提供盡可能多的信息,以便他可以避免或處理異常。

因此,請確保向Javadoc 添加@throws聲明并描述可能導(dǎo)致異常的情況。

/**
 * This method does something extremely useful ...
 *
 * @param input
 * @throws MyBusinessException if ... happens
 */
public void doSomething(String input) throws MyBusinessException {
 ...
}

4.使用描述信息拋出異常

這種最佳實(shí)踐背后的想法類似于前兩種實(shí)踐。但是這次,你不向調(diào)用方提供有關(guān)方法的信息。每個(gè)必須了解在日志文件或監(jiān)視工具中拋出異常時(shí)發(fā)生了什么的人都會讀取異常的消息。

因此,它應(yīng)該盡可能準(zhǔn)確地描述問題,并提供最相關(guān)的信息來理解異常事件。

別誤會我的意思; 你不應(yīng)該寫一段文字。但是你應(yīng)該用1-2個(gè)簡短的句子來解釋這個(gè)例外的原因。這有助于你的運(yùn)營團(tuán)隊(duì)了解問題的嚴(yán)重性,還可以讓你更輕松地分析任何服務(wù)事件。

如果拋出一個(gè)特定的異常,它的類名很可能已經(jīng)描述了那種錯(cuò)誤。因此,你無需提供大量其他信息。一個(gè)很好的例子是NumberFormatException。它會被類java.lang.Long的構(gòu)造函數(shù)拋出,當(dāng)你以錯(cuò)誤的格式提供String參數(shù)。

try {
 new Long("xyz");
} catch (NumberFormatException e) {
 log.error(e);
}

NumberFormatException類的名稱已經(jīng)告訴你問題的類型。它的消息只需要提供導(dǎo)致問題的輸入字符串。如果異常類的名稱不具有表現(xiàn)力,則需要在消息中提供所需的信息。

17:17:26,386 ERROR TestExceptionHandling:52 - java.lang.NumberFormatException: For input string: "xyz"

5.優(yōu)先捕獲最具體的異常

大多數(shù)IDE都可以幫助你實(shí)現(xiàn)這一最佳實(shí)踐。當(dāng)你嘗試首先捕獲不太具體的異常時(shí),它們提示無法訪問的代碼塊。

問題是只有匹配異常的第一個(gè)catch塊才會被執(zhí)行。因此,如果首先捕獲IllegalArgumentException,則永遠(yuǎn)不會到達(dá)應(yīng)該處理更具體的NumberFormatException的catch塊,因?yàn)樗荌llegalArgumentException的子類。

始終優(yōu)先捕獲最具體的異常類,并將不太具體的catch塊添加到列表的末尾。

你可以在以下代碼段中看到此類try-catch語句的示例。第一個(gè)catch塊處理所有的NumberFormatException,第二個(gè)處理所有不是NumberFormatException的IllegalArgumentException 異常。

public void catchMostSpecificExceptionFirst() {
 try {
 doSomething("A message");
 } catch (NumberFormatException e) {
 log.error(e);
 } catch (IllegalArgumentException e) {
 log.error(e)
 }
}

6.Don't Catch Throwable

Throwable是所有異常和錯(cuò)誤的超類。你可以在catch子句中使用它,但你永遠(yuǎn)不應(yīng)該這樣做!

如果在catch子句中使用Throwable,它不僅會捕獲所有異常; 它還會捕獲所有錯(cuò)誤。JVM拋出錯(cuò)誤以指示應(yīng)用程序無法處理的嚴(yán)重問題。典型的例子是OutOfMemoryError或StackOverflowError。兩者都是由應(yīng)用程序無法控制的情況引起的,無法處理。

所以,最好不要抓住Throwable,除非你完全確定你處于一個(gè)特殊情況,你可以或者需要處理錯(cuò)誤。

public void doNotCatchThrowable() {
 try {
 // do something
 } catch (Throwable t) {
 // don't do this!
 }
}

7.Don't Ignore Exceptions

你是否曾經(jīng)分析過只有用例第一部分被執(zhí)行的錯(cuò)誤報(bào)告?

這通常是由忽略的異常引起的。開發(fā)人員可能非常確定它永遠(yuǎn)不會被拋出并添加了一個(gè)不處理或記錄它的catch塊。當(dāng)你找到這個(gè)代碼塊時(shí),你很可能甚至?xí)l(fā)現(xiàn)一個(gè)著名的“This will never happen

”的評論。

public void doNotIgnoreExceptions() {
 try {
 // do something
 } catch (NumberFormatException e) {
 // this will never happen
 }
}

好吧,你可能正在分析一個(gè)不可能發(fā)生的問題。

所以,請永遠(yuǎn)不要忽視異常。你不知道代碼將來會如何變化。有人可能會刪除阻止異常事件的驗(yàn)證而不會認(rèn)識到這會產(chǎn)生問題。或者拋出異常的代碼會被更改,現(xiàn)在拋出同一個(gè)類的多個(gè)異常,并且調(diào)用代碼不會阻止所有這些異常。

你至少應(yīng)該寫一條日志消息,告訴大家不可思議的事情剛剛發(fā)生,而且有人需要檢查它。

public void logAnException() {
 try {
 // do something
 } catch (NumberFormatException e) {
 log.error("This should never happen: " + e);
 }
}

8.Don't Log and Throw

這可能是此列表中最常被忽略的最佳做法。你可以找到許多代碼片段,甚至是catch,log和重新throw異常的庫。

try {
 new Long("xyz");
} catch (NumberFormatException e) {
 log.error(e);
 throw e;
}

在發(fā)生異常時(shí)記錄異??赡軙杏X很直接,然后重新拋出它以便調(diào)用者可以適當(dāng)?shù)靥幚硭?。但它會為同一個(gè)異常寫出多條錯(cuò)誤消息。

17:44:28,945 ERROR TestExceptionHandling:65 - java.lang.NumberFormatException: For input string: "xyz"

Exception in thread "main" java.lang.NumberFormatException: For input string: "xyz"

at java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)

at java.lang.Long.parseLong(Long.java:589)

at java.lang.Long.(Long.java:965)

at com.stackify.example.TestExceptionHandling.logAndThrowException(TestExceptionHandling.java:63)

at com.stackify.example.TestExceptionHandling.main(TestExceptionHandling.java:58)

其他消息也不添加任何信息。如最佳實(shí)踐#4中所述,異常消息應(yīng)描述異常事件。堆棧跟蹤告訴你拋出異常的類,方法和行。

如果需要添加其他信息,則應(yīng)捕獲異常并將其包裝在自定義異常中。但請務(wù)必遵循最佳做法9。

public void wrapException(String input) throws MyBusinessException {
 try {
 // do something
 } catch (NumberFormatException e) {
 throw new MyBusinessException("A message that describes the error.", e);
 }
}

因此,如果你想要處理它,只捕獲異常。否則,在方法簽名中指定它并讓調(diào)用者處理它。

9.在沒有消費(fèi)的情況下包裝異常

有時(shí)候捕獲標(biāo)準(zhǔn)異常并將其包裝成自定義異常會更好。此類異常的典型示例是應(yīng)用程序或框架特定的業(yè)務(wù)異常。這允許你添加其他信息,還可以為異常類實(shí)現(xiàn)特殊處理。

執(zhí)行此操作時(shí),請確保將原始異常設(shè)置為cause。該異常類提供了接受一個(gè)特定的構(gòu)造方法的Throwable作為參數(shù)。否則,你將丟失原始異常的堆棧跟蹤和消息,這將導(dǎo)致難以分析導(dǎo)致異常的異常事件。

public void wrapException(String input) throws MyBusinessException {
 try {
 // do something
 } catch (NumberFormatException e) {
 throw new MyBusinessException("A message that describes the error.", e);
 }
}

總結(jié)

正如所看到的,當(dāng)你拋出或捕獲異常時(shí),你應(yīng)該考慮許多不同的事情。其中大多數(shù)都旨在提高代碼的可讀性或API的可用性。

異常通常同時(shí)是錯(cuò)誤處理機(jī)制和通信媒介。因此,您應(yīng)該確保與同事討論要應(yīng)用的最佳實(shí)踐和規(guī)則,以便每個(gè)人都能理解通用概念并以相同的方式使用它們。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Spring boot2X負(fù)載均衡和反向代理實(shí)現(xiàn)過程解析

    Spring boot2X負(fù)載均衡和反向代理實(shí)現(xiàn)過程解析

    這篇文章主要介紹了Spring boot2X負(fù)載均衡和反向代理實(shí)現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-12-12
  • 三步輕松搭建springMVC框架

    三步輕松搭建springMVC框架

    這篇文章主要教大家三步輕松搭建springMVC框架,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • 淺聊一下Spring中Bean的配置細(xì)節(jié)

    淺聊一下Spring中Bean的配置細(xì)節(jié)

    我們知道,當(dāng)寫完一個(gè)普通的 Java 類后,想讓 Spring IoC 容器在創(chuàng)建類的實(shí)例對象時(shí)使用構(gòu)造方法完成實(shí)例對象的依賴注入,那么就需要在配置元數(shù)據(jù)中寫好類的 Bean 定義,包括各種標(biāo)簽的屬性。所以本文我們來說說這其中的配置細(xì)節(jié),需要的朋友可以參考下
    2023-07-07
  • Java中的內(nèi)部類使用詳情

    Java中的內(nèi)部類使用詳情

    說起內(nèi)部類這個(gè)詞,想必很多人都不陌生,但是又會覺得不熟悉。原因是平時(shí)編寫代碼時(shí)可能用到的場景不多,用得最多的是在有事件監(jiān)聽的情況下,并且即使用到也很少去總結(jié)內(nèi)部類的用法。今天我們就來一探究竟
    2022-03-03
  • 詳解java動態(tài)代理模式

    詳解java動態(tài)代理模式

    這篇文章主要為大家詳細(xì)介紹了java動態(tài)代理模式,總結(jié)一下代理模式,以及jdk,cglib代理模式用法,來理解代理模式,感興趣的小伙伴們可以參考一下
    2016-02-02
  • 詳解IDEA啟動多個(gè)微服務(wù)的配置方法

    詳解IDEA啟動多個(gè)微服務(wù)的配置方法

    這篇文章主要介紹了詳解IDEA啟動多個(gè)微服務(wù)的配置方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-01-01
  • java使用itext導(dǎo)出PDF文本絕對定位(實(shí)現(xiàn)方法)

    java使用itext導(dǎo)出PDF文本絕對定位(實(shí)現(xiàn)方法)

    下面小編就為大家?guī)硪黄猨ava使用itext導(dǎo)出PDF文本絕對定位(實(shí)現(xiàn)方法)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-06-06
  • eclipse下搭建hibernate5.0環(huán)境的步驟(圖文)

    eclipse下搭建hibernate5.0環(huán)境的步驟(圖文)

    這篇文章主要介紹了eclipse下搭建hibernate5.0環(huán)境的步驟(圖文),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-05-05
  • Java函數(shù)式編程(九):Comparator

    Java函數(shù)式編程(九):Comparator

    這篇文章主要介紹了Java函數(shù)式編程(九):Comparator,本文是系列文章的第9篇,其它文章請參閱本文底部的相關(guān)文章,需要的朋友可以參考下
    2014-09-09
  • 關(guān)于springboot配置文件密文解密方式

    關(guān)于springboot配置文件密文解密方式

    這篇文章主要介紹了關(guān)于springboot配置文件密文解密方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08

最新評論