如何優雅的設計 Java 異常

2020-12-12 酷扯兒

本文轉載自【微信公眾號:java進階架構師,ID:java_jiagoushi】經微信公眾號授權轉載,如需轉載與原文作者聯繫

導語

異常處理是程序開發中必不可少操作之一,但如何正確優雅的對異常進行處理確是一門學問,筆者根據自己的開發經驗來談一談我是如何對異常進行處理的。

由於本文只作一些經驗之談,不涉及到基礎知識部分,如果讀者對異常的概念還很模糊,請先查看基礎知識。

如何選擇異常類型

異常的類別

正如我們所知道的,java中的異常的超類是java.lang.Throwable(後文省略為Throwable),它有兩個比較重要的子類,java.lang.Exception(後文省略為Exception)和java.lang.Error(後文省略為Error),其中Error由JVM虛擬機進行管理,如我們所熟知的OutOfMemoryError異常等,所以我們本文不關注Error異常,那麼我們細說一下Exception異常。

Exception異常有個比較重要的子類,叫做RuntimeException。我們將RuntimeException或其他繼承自RuntimeException的子類稱為非受檢異常(unchecked Exception),其他繼承自Exception異常的子類稱為受檢異常(checked Exception)。本文重點來關注一下受檢異常和非受檢異常這兩種異常。

如何選擇異常

從筆者的開發經驗來看,如果在一個應用中,需要開發一個方法(如某個功能的service方法),這個方法如果中間可能出現異常,那麼你需要考慮這個異常出現之後是否調用者可以處理,並且你是否希望調用者進行處理,如果調用者可以處理,並且你也希望調用者進行處理,那麼就要拋出受檢異常,提醒調用者在使用你的方法時,考慮到如果拋出異常時如果進行處理,相似的,如果在寫某個方法時,你認為這是個偶然異常,理論上說,你覺得運行時可能會碰到什麼問題,而這些問題也許不是必然發生的,也不需要調用者顯示的通過異常來判斷業務流程操作的,那麼這時就可以使用一個RuntimeException這樣的非受檢異常.

好了,估計我上邊說的這段話,你讀了很多遍也依然覺得晦澀了。

那麼,請跟著我的思路,在慢慢領會一下。

什麼時候才需要拋異常

首先我們需要了解一個問題,什麼時候才需要拋異常?異常的設計是方便給開發者使用的,但不是亂用的,筆者對於什麼時候拋異常這個問題也問了很多朋友,能給出準確答案的確實不多。其實這個問題很簡單,如果你覺得某些」問題」解決不了了,那麼你就可以拋出異常了。比如,你在寫一個service,其中在寫到某段代碼處,你發現可能會產生問題,那麼就請拋出異常吧,相信我,你此時拋出異常將是一個最佳時機。

應該拋出怎樣的異常

了解完了什麼時候才需要拋出異常後,我們再思考一個問題,真的當我們拋出異常時,我們應該選用怎樣的異常呢?究竟是受檢異常還是非受檢異常呢(RuntimeException)呢?我來舉例說明一下這個問題,先從受檢異常說起,比如說有這樣一個業務邏輯,需要從某文件中讀取某個數據,這個讀取操作可能是由於文件被刪除等其他問題導致無法獲取從而出現讀取錯誤,那麼就要從redis或mysql資料庫中再去獲取此數據,參考如下代碼,getKey(Integer)為入口程序.

public String getKey(Integer key){String value; try { InputStream inputStream = getFiles("/file/nofile"); //接下來從流中讀取key的value指 value = ...; } catch (Exception e) { //如果拋出異常將從mysql或者redis進行取之 value = ...; }}public InputStream getFiles(String path) throws Exception { File file = new File(path); InputStream inputStream = null; try { inputStream = new BufferedInputStream(new FileInputStream(file)); } catch (FileNotFoundException e) { throw new Exception("I/O讀取錯誤",e.getCause()); } return inputStream;}

ok,看了以上代碼以後,你也許心中有一些想法,原來受檢異常可以控制義務邏輯,對,沒錯,通過受檢異常真的可以控制業務邏輯,但是切記不要這樣使用,我們應該合理的拋出異常,因為程序本身才是流程,異常的作用僅僅是當你進行不下去的時候找到的一個藉口而已,它並不能當成控制程序流程的入口或出口,如果這樣使用的話,是在將異常的作用擴大化,這樣將會導致代碼複雜程度的增加,耦合性會提高,代碼可讀性降低等問題。那麼就一定不要使用這樣的異常嗎?其實也不是,在真的有這樣的需求的時候,我們可以這樣使用,只是切記,不要把它真的當成控制流程的工具或手段。那麼究竟什麼時候才要拋出這樣的異常呢?要考慮,如果調用者調用出錯後,一定要讓調用者對此錯誤進行處理才可以,滿足這樣的要求時,我們才會考慮使用受檢異常。

接下來,我們來看一下非受檢異常呢(RuntimeException),對於RuntimeException這種異常,我們其實很多見,比如java.lang.NullPointerException/java.lang.IllegalArgumentException等,那麼這種異常我們時候拋出呢?當我們在寫某個方法的時候,可能會偶然遇到某個錯誤,我們認為這個問題時運行時可能為發生的,並且理論上講,沒有這個問題的話,程序將會正常執行的時候,它不強制要求調用者一定要捕獲這個異常,此時拋出RuntimeException異常,舉個例子,當傳來一個路徑的時候,需要返回一個路徑對應的File對象:

public void test() {myTest.getFiles("");}public File getFiles(String path) { if(null == path || "".equals(path)){ throw new NullPointerException("路徑不能為空!"); } File file = new File(path); return file;}

上述例子表明,如果調用者調用getFiles(String)的時候如果path是空,那麼就拋出空指針異常(它是RuntimeException的子類),調用者不用顯示的進行try…catch…操作進行強制處理.這就要求調用者在調用這樣的方法時先進行驗證,避免發生RuntimeException.如下:

應該選用哪種異常

通過以上的描述和舉例,可以總結出一個結論,RuntimeException異常和受檢異常之間的區別就是:是否強制要求調用者必須處理此異常,如果強制要求調用者必須進行處理,那麼就使用受檢異常,否則就選擇非受檢異常(RuntimeException)。一般來講,如果沒有特殊的要求,我們建議使用RuntimeException異常。

場景介紹和技術選型

架構描述

正如我們所知,傳統的項目都是以MVC框架為基礎進行開發的,本文主要從使用restful風格接口的設計來體驗一下異常處理的優雅。

我們把關注點放在restful的api層(和web中的controller層類似)和service層,研究一下在service中如何拋出異常,然後api層如何進行捕獲並且轉化異常。

使用的技術是:spring-boot,jpa(hibernate),mysql,如果對這些技術不是太熟悉,讀者需要自行閱讀相關材料。

業務場景描述

選擇一個比較簡單的業務場景,以電商中的收貨地址管理為例,用戶在移動端進行購買商品時,需要進行收貨地址管理,在項目中,提供一些給移動端進行訪問的api接口,如:添加收貨地址,刪除收貨地址,更改收貨地址,默認收貨地址設置,收貨地址列表查詢,單個收貨地址查詢等接口。

構建約束條件

ok,這個是設置好的一個很基本的業務場景,當然,無論什麼樣的api操作,其中都包含一些規則:

添加收貨地址:

入參:

用戶id收貨地址實體信息

約束:

用戶id不能為空,且此用戶確實是存在 的收貨地址的必要欄位不能為 空如果用戶還沒有收貨地址,當此收貨地址創建時設置成默認收貨地址 —

刪除收貨地址:

入參:

用戶id收貨地址id

約束:

用戶id不能為空,且此用戶確實是存在的收貨地址不能為空,且此收貨地址確實是存在的判斷此收貨地址是否是用戶的收貨地址判斷此收貨地址是否為默認收貨地址,如果是默認收貨地址,那麼不能進行刪除

更改收貨地址:

入參:

用戶id收貨地址id

約束:

用戶id不能為空,且此用戶確實是存在的收貨地址不能為空,且此收貨地址確實是存在的判斷此收貨地址是否是用戶的收貨地址

默認地址設置:

入參:

用戶id收貨地址id

約束:

用戶id不能為空,且此用戶確實是存在的收貨地址不能為空,且此收貨地址確實是存在的判斷此收貨地址是否是用戶的收貨地址

收貨地址列表查詢:

入參:

用戶id

約束:

用戶id不能為空,且此用戶確實是存在的

單個收貨地址查詢:

入參:

用戶id收貨地址id

約束:

用戶id不能為空,且此用戶確實是存在的收貨地址不能為空,且此收貨地址確實是存在的判斷此收貨地址是否是用戶的收貨地址

約束判斷和技術選型

對於上述列出的約束條件和功能列表,我選擇幾個比較典型的異常處理場景進行分析:添加收貨地址,刪除收貨地址,獲取收貨地址列表。

那麼應該有哪些必要的知識儲備呢,讓我們看一下收貨地址這個功能:

添加收貨地址中需要對用戶id和收貨地址實體信息就行校驗,那麼對於非空的判斷,我們如何進行工具的選擇呢?傳統的判斷如下:

/*** 添加地址 * @param uid * @param address * @return */public Address addAddress(Integer uid,Address address){ if(null != uid){ //進行處理.. } return null;}

上邊的例子,如果只判斷uid為空還好,如果再去判斷address這個實體中的某些必要屬性是否為空,在欄位很多的情況下,這無非是災難性的。

那我們應該怎麼進行這些入參的判斷呢,給大家介紹兩個知識點:

Guava中的Preconditions類實現了很多入參方法的判斷jsr 303的validation規範(目前實現比較全的是hibernate實現的hibernate-validator)

如果使用了這兩種推薦技術,那麼入參的判斷會變得簡單很多。推薦大家多使用這些成熟的技術和jar工具包,他可以減少很多不必要的工作量。我們只需要把重心放到業務邏輯上。而不會因為這些入參的判斷耽誤更多的時間。

如何優雅的設計java異常

domain介紹

根據項目場景來看,需要兩個domain模型,一個是用戶實體,一個是地址實體.

Address domain如下:

@Entity@Datapublic class Address {@Id @GeneratedValue private Integer id; private String province;//省 private String city;//市 private String county;//區 private Boolean isDefault;//是否是默認地址 @ManyToOne(cascade={CascadeType.ALL}) @JoinColumn(name="uid") private User user;}

User domain如下:

@Entity@Datapublic class User {@Id @GeneratedValue private Integer id; private String name;//姓名 @OneToMany(cascade= CascadeType.ALL,mappedBy="user",fetch = FetchType.LAZY) private Set<Address> addresses;}

ok,上邊是一個模型關係,用戶-收貨地址的關係是1-n的關係。上邊的@Data是使用了一個叫做lombok的工具,它自動生成了Setter和Getter等方法,用起來非常方便,感興趣的讀者可以自行了解一下。

dao介紹

數據連接層,我們使用了spring-data-jpa這個框架,它要求我們只需要繼承框架提供的接口,並且按照約定對方法進行取名,就可以完成我們想要的資料庫操作。

用戶資料庫操作如下:

@Repositorypublic interface IUserDao extends JpaRepository<User,Integer> {}

收貨地址操作如下:

@Repositorypublic interface IAddressDao extends JpaRepository<Address,Integer> {}

正如讀者所看到的,我們的DAO只需要繼承JpaRepository,它就已經幫我們完成了基本的CURD等操作,如果想了解更多關於spring-data的這個項目,請參考一下spring的官方文檔,它比不方案我們對異常的研究。

Service異常設計

ok,終於到了我們的重點了,我們要完成service一些的部分操作:添加收貨地址,刪除收貨地址,獲取收貨地址列表.

首先看我的service接口定義:

public interface IAddressService {/*** 創建收貨地址 * @param uid * @param address * @return */Address createAddress(Integer uid,Address address);/** * 刪除收貨地址 * @param uid * @param aid */void deleteAddress(Integer uid,Integer aid);/** * 查詢用戶的所有收貨地址 * @param uid * @return */List<Address> listAddresses(Integer uid);}

我們來關注一下實現:

添加收貨地址

首先再來看一下之前整理的約束條件:

入參:

用戶id收貨地址實體信息

約束:

用戶id不能為空,且此用戶確實是存在的收貨地址的必要欄位不能為空如果用戶還沒有收貨地址,當此收貨地址創建時設置成默認收貨地址

先看以下代碼實現:

@Overridepublic Address createAddress(Integer uid, Address address) {//============ 以下為約束條件 ============== //1.用戶id不能為空,且此用戶確實是存在的 Preconditions.checkNotNull(uid); User user = userDao.findOne(uid); if(null == user){ throw new RuntimeException("找不到當前用戶!"); } //2.收貨地址的必要欄位不能為空 BeanValidators.validateWithException(validator, address); //3.如果用戶還沒有收貨地址,當此收貨地址創建時設置成默認收貨地址 if(ObjectUtils.isEmpty(user.getAddresses())){ address.setIsDefault(true); } //============ 以下為正常執行的業務邏輯 ============== address.setUser(user); Address result = addressDao.save(address); return result;}

其中,已經完成了上述所描述的三點約束條件,當三點約束條件都滿足時,才可以進行正常的業務邏輯,否則將拋出異常(一般在此處建議拋出運行時異常-RuntimeException)。

介紹以下以上我所用到的技術:

1. Preconfitions.checkNotNull(T t)這個是使用Guava中的com.google.common.base.Preconditions進行判斷的,因為service中用到的驗證較多,所以建議將Preconfitions改成靜態導入的方式:

import static com.google.common.base.Preconditions.checkNotNull;

當然Guava的github中的說明也建議我們這樣使用。

2. BeanValidators.validateWithException(validator, address);

這個使用了hibernate實現的jsr 303規範來做的,需要傳入一個validator和一個需要驗證的實體,那麼validator是如何獲取的呢,如下:

@Configurationpublic class BeanConfigs {@Beanpublic javax.validation.Validator getValidator(){return new LocalValidatorFactoryBean();}}

他將獲取一個Validator對象,然後我們在service中進行注入便可以使用了:

@Autowired

private Validator validator ;

那麼BeanValidators這個類是如何實現的?其實實現方式很簡單,只要去判斷jsr 303的標註註解就ok了。

那麼jsr 303的註解寫在哪裡了呢?當然是寫在address實體類中了:

@Entity@Setter@Getterpublic class Address {@Id@GeneratedValue private Integer id; @NotNullprivate String province;//省@NotNullprivate String city;//市@NotNullprivate String county;//區private Boolean isDefault = false;//是否是默認地址@ManyToOne(cascade={CascadeType.ALL})@JoinColumn(name="uid")private User user;}

寫好你需要的約束條件來進行判斷,如果合理的話,才可以進行業務操作,從而對資料庫進行操作。

這塊的驗證是必須的,一個最主要的原因是:這樣的驗證可以避免髒數據的插入。如果讀者有正式上線的經驗的話,就可以理解這樣的一個事情,任何的代碼錯誤都可以容忍和修改,但是如果出現了髒數據問題,那麼它有可能是一個毀滅性的災難。程序的問題可以修改,但是髒數據的出現有可能無法恢復。所以這就是為什麼在service中一定要判斷好約束條件,再進行業務邏輯操作的原因了。

此處的判斷為業務邏輯判斷,是從業務角度來進行篩選判斷的,除此之外,有可能在很多場景中都會有不同的業務條件約束,只需要按照要求來做就好。

對於約束條件的總結如下:

基本判斷約束(null值等基本判斷)實體屬性約束(滿足jsr 303等基礎判斷)業務條件約束(需求提出的不同的業務約束)

當這個三點都滿足時,才可以進行下一步操作

ok,基本介紹了如何做一個基礎的判斷,那麼再回到異常的設計問題上,上述代碼已經很清楚的描述如何在適當的位置合理的判斷一個異常了,那麼如何合理的拋出異常呢?

只拋出RuntimeException就算是優雅的拋出異常嗎?當然不是,對於service中的拋出異常,筆者認為大致有兩種拋出的方法:

拋出帶狀態碼RumtimeException異常拋出指定類型的RuntimeException異常

相對這兩種異常的方式進行結束,第一種異常指的是我所有的異常都拋RuntimeException異常,但是需要帶一個狀態碼,調用者可以根據狀態碼再去查詢究竟service拋出了一個什麼樣的異常。

第二種異常是指在service中拋出什麼樣的異常就自定義一個指定的異常錯誤,然後在進行拋出異常。

一般來講,如果系統沒有別的特殊需求的時候,在開發設計中,建議使用第二種方式。但是比如說像基礎判斷的異常,就可以完全使用guava給我們提供的類庫進行操作。jsr 303異常也可以使用自己封裝好的異常判斷類進行操作,因為這兩種異常都是屬於基礎判斷,不需要為它們指定特殊的異常。但是對於第三點義務條件約束判斷拋出的異常,就需要拋出指定類型的異常了。

對於

throw new RuntimeException("找不到當前用戶!");

定義一個特定的異常類來進行這個義務異常的判斷:

public class NotFindUserException extends RuntimeException {public NotFindUserException() {super("找不到此用戶");}public NotFindUserException(String message) { super(message);}}

然後將此處改為:

throw new NotFindUserException("找不到當前用戶!");

or

throw new NotFindUserException();

ok,通過以上對service層的修改,代碼更改如下:

@Overridepublic Address createAddress(Integer uid, Address address) {//============ 以下為約束條件 ============== //1.用戶id不能為空,且此用戶確實是存在的 checkNotNull(uid); User user = userDao.findOne(uid); if(null == user){ throw new NotFindUserException("找不到當前用戶!"); } //2.收貨地址的必要欄位不能為空 BeanValidators.validateWithException(validator, address); //3.如果用戶還沒有收貨地址,當此收貨地址創建時設置成默認收貨地址 if(ObjectUtils.isEmpty(user.getAddresses())){ address.setIsDefault(true); } //============ 以下為正常執行的業務邏輯 ============== address.setUser(user); Address result = addressDao.save(address); return result;}

這樣的service就看起來穩定性和理解性就比較強了。

刪除收貨地址:

入參:

用戶id收貨地址id

約束:

用戶id不能為空,且此用戶確實是存在的收貨地址不能為空,且此收貨地址確實是存在的判斷此收貨地址是否是用戶的收貨地址判斷此收貨地址是否為默認收貨地址,如果是默認收貨地址,那麼不能進行刪除

它與上述添加收貨地址類似,故不再贅述,delete的service設計如下:

@Overridepublic void deleteAddress(Integer uid, Integer aid) {//============ 以下為約束條件 ============== //1.用戶id不能為空,且此用戶確實是存在的 checkNotNull(uid); User user = userDao.findOne(uid); if(null == user){ throw new NotFindUserException(); } //2.收貨地址不能為空,且此收貨地址確實是存在的 checkNotNull(aid); Address address = addressDao.findOne(aid); if(null == address){ throw new NotFindAddressException(); } //3.判斷此收貨地址是否是用戶的收貨地址 if(!address.getUser().equals(user)){ throw new NotMatchUserAddressException(); } //4.判斷此收貨地址是否為默認收貨地址,如果是默認收貨地址,那麼不能進行刪除 if(address.getIsDefault()){ throw new DefaultAddressNotDeleteException(); } //============ 以下為正常執行的業務邏輯 ============== addressDao.delete(address);}

設計了相關的四個異常類:NotFindUserException,NotFindAddressException,NotMatchUserAddressException,DefaultAddressNotDeleteException.根據不同的業務需求拋出不同的異常。

獲取收貨地址列表:

入參:

用戶id

約束:

用戶id不能為空,且此用戶確實是存在的

代碼如下:

@Overridepublic List<Address> listAddresses(Integer uid) {//============ 以下為約束條件 ============== //1.用戶id不能為空,且此用戶確實是存在的 checkNotNull(uid); User user = userDao.findOne(uid); if(null == user){ throw new NotFindUserException(); } //============ 以下為正常執行的業務邏輯 ============== User result = userDao.findOne(uid); return result.getAddresses();}

api異常設計

大致有兩種拋出的方法:

拋出帶狀態碼RumtimeException異常拋出指定類型的RuntimeException異常

這個是在設計service層異常時提到的,通過對service層的介紹,我們在service層拋出異常時選擇了第二種拋出的方式,不同的是,在api層拋出異常我們需要使用這兩種方式進行拋出:要指定api異常的類型,並且要指定相關的狀態碼,然後才將異常拋出,這種異常設計的核心是讓調用api的使用者更能清楚的了解發生異常的詳細信息,除了拋出異常外,我們還需要將狀態碼對應的異常詳細信息以及異常有可能發生的問題製作成一個對應的表展示給用戶,方便用戶的查詢。(如github提供的api文檔,微信提供的api文檔等),還有一個好處:如果用戶需要自定義提示消息,可以根據返回的狀態碼進行提示的修改。

api驗證約束

首先對於api的設計來說,需要存在一個dto對象,這個對象負責和調用者進行數據的溝通和傳遞,然後dto->domain在傳給service進行操作,這一點一定要注意,第二點,除了說道的service需要進行基礎判斷(null判斷)和jsr 303驗證以外,同樣的,api層也需要進行相關的驗證,如果驗證不通過的話,直接返回給調用者,告知調用失敗,不應該帶著不合法的數據再進行對service的訪問,那麼讀者可能會有些迷惑,不是service已經進行驗證了,為什麼api層還需要進行驗證麼?這裡便設計到了一個概念:編程中的墨菲定律,如果api層的數據驗證疏忽了,那麼有可能不合法數據就帶到了service層,進而講髒數據保存到了資料庫。

所以縝密編程的核心是:永遠不要相信收到的數據是合法的。

api異常設計

設計api層異常時,正如我們上邊所說的,需要提供錯誤碼和錯誤信息,那麼可以這樣設計,提供一個通用的api超類異常,其他不同的api異常都繼承自這個超類:

public class ApiException extends RuntimeException {protected Long errorCode ;protected Object data ;public ApiException(Long errorCode,String message,Object data,Throwable e){super(message,e); this.errorCode = errorCode ; this.data = data ;}public ApiException(Long errorCode,String message,Object data){ this(errorCode,message,data,null);}public ApiException(Long errorCode,String message){ this(errorCode,message,null,null);}public ApiException(String message,Throwable e){ this(null,message,null,e);}public ApiException(){}public ApiException(Throwable e){ super(e);}public Long getErrorCode() { return errorCode;}public void setErrorCode(Long errorCode) { this.errorCode = errorCode;}public Object getData() { return data;}public void setData(Object data) { this.data = data;}}

然後分別定義api層異常:ApiDefaultAddressNotDeleteException,ApiNotFindAddressException,ApiNotFindUserException,ApiNotMatchUserAddressException。

以默認地址不能刪除為例:

public class ApiDefaultAddressNotDeleteException extends ApiException {public ApiDefaultAddressNotDeleteException(String message) {super(AddressErrorCode.DefaultAddressNotDeleteErrorCode, message, null);}}

AddressErrorCode.DefaultAddressNotDeleteErrorCode就是需要提供給調用者的錯誤碼。錯誤碼類如下:

public abstract class AddressErrorCode {public static final Long DefaultAddressNotDeleteErrorCode = 10001L;//默認地址不能刪除 public static final Long NotFindAddressErrorCode = 10002L;//找不到此收貨地址 public static final Long NotFindUserErrorCode = 10003L;//找不到此用戶 public static final Long NotMatchUserAddressErrorCode = 10004L;//用戶與收貨地址不匹配}

ok,那麼api層的異常就已經設計完了,在此多說一句,AddressErrorCode錯誤碼類存放了可能出現的錯誤碼,更合理的做法是把他放到配置文件中進行管理。

api處理異常

api層會調用service層,然後來處理service中出現的所有異常,首先,需要保證一點,一定要讓api層非常輕,基本上做成一個轉發的功能就好(接口參數,傳遞給service參數,返回給調用者數據,這三個基本功能),然後就要在傳遞給service參數的那個方法調用上進行異常處理。

此處僅以添加地址為例:

@Autowiredprivate IAddressService addressService;/*** 添加收貨地址 * @param addressDTO * @return */@RequestMapping(method = RequestMethod.POST)public AddressDTO add(@Valid @RequestBody AddressDTO addressDTO){ Address address = new Address(); BeanUtils.copyProperties(addressDTO,address); Address result; try { result = addressService.createAddress(addressDTO.getUid(), address); }catch (NotFindUserException e){ throw new ApiNotFindUserException("找不到該用戶"); }catch (Exception e){//未知錯誤 throw new ApiException(e); } AddressDTO resultDTO = new AddressDTO(); BeanUtils.copyProperties(result,resultDTO); resultDTO.setUid(result.getUser().getId()); return resultDTO;}

這裡的處理方案是調用service時,判斷異常的類型,然後將任何service異常都轉化成api異常,然後拋出api異常,這是常用的一種異常轉化方式。相似刪除收貨地址和獲取收貨地址也類似這樣處理,在此,不在贅述。

api異常轉化

已經講解了如何拋出異常和何如將service異常轉化為api異常,那麼轉化成api異常直接拋出是否就完成了異常處理呢? 答案是否定的,當拋出api異常後,我們需要把api異常返回的數據(json or xml)讓用戶看懂,那麼需要把api異常轉化成dto對象(ErrorDTO),看如下代碼:

@ControllerAdvice(annotations = RestController.class)class ApiExceptionHandlerAdvice {/*** Handle exceptions thrown by handlers. */@ExceptionHandler(value = Exception.class)@ResponseBodypublic ResponseEntity<ErrorDTO> exception(Exception exception,HttpServletResponse response) { ErrorDTO errorDTO = new ErrorDTO(); if(exception instanceof ApiException){//api異常 ApiException apiException = (ApiException)exception; errorDTO.setErrorCode(apiException.getErrorCode()); }else{//未知異常 errorDTO.setErrorCode(0L); } errorDTO.setTip(exception.getMessage()); ResponseEntity<ErrorDTO> responseEntity = new ResponseEntity<>(errorDTO,HttpStatus.valueOf(response.getStatus())); return responseEntity;}@Setter@Getterclass ErrorDTO{ private Long errorCode; private String tip;}}

ok,這樣就完成了api異常轉化成用戶可以讀懂的DTO對象了,代碼中用到了@ControllerAdvice,這是spring MVC提供的一個特殊的切面處理。

當調用api接口發生異常時,用戶也可以收到正常的數據格式了,比如當沒有用戶(uid為2)時,卻為這個用戶添加收貨地址,postman(Google plugin 用於模擬http請求)之後的數據:

{"errorCode": 10003, "tip": "找不到該用戶"}

總結

本文只從如何設計異常作為重點來講解,涉及到的api傳輸和service的處理,還有待優化,比如api接口訪問需要使用https進行加密,api接口需要OAuth2.0授權或api接口需要籤名認證等問題,文中都未曾提到,本文的重心在於異常如何處理,所以讀者只需關注涉及到異常相關的問題和處理方式就可以了。希望本篇文章對你理解異常有所幫助。

相關焦點

  • Java:如何更優雅地處理空值?
    尤其是對接口的抽象,它在設計和開發中佔很大的比重,我們在開發時希望儘量面向接口編程。那如何約束入參呢?Optioanl的正確使用Optional如此強大,它表達了計算機最原始的特性(0 or 1),那它如何正確的被使用呢!
  • java如何快速入門?
    java如何快速入門正確掌握Java的基本知識由於Java為開發人員提供了如此多的特性和選項,人們有時會被分散注意力在太短的時間內學習了太多的東西java如何快速入門真正弄懂你敲出的代碼是做什麼的只要你理解了代碼背後的思想,算法和整個編譯過程就會顯得非常有意義。
  • Java Socket網絡編程, 五個常見的異常發生的原因以及對應的解決方案
    空調WiFi西瓜,葛優同款沙發夕陽西下,我打開電腦學 java ……在Java網絡編程中,我們經常性的會碰到一些異常,有些異常是我們反覆碰見的,下面整理幾條常見的異常,供大家參考交流。1.java.net.
  • 你必須掌握的 21 個 Java 核心技術!
    這個知識點是最最基本的java開發者需要掌握的,第一個肯定是教你如何在命令行中執行java程序,但是很多人一旦把java學完了,IDE用上了,就把這個都忘了。為什麼強調要知道這個呢,知道了java最純粹的啟動方式之後,你才能在啟動出問題的時候,去分析當時啟動的目錄多少,執行命名如何,參數如何,是否有缺失等。這樣有利於你真正開發中去解決那些奇奇怪怪的可能和環境相關的問題。
  • 實戰系列:使用Java8 Optional類優雅解決空指針問題
    Java8吸收其他語言的精髓帶來了函數式編程,lambda表達式,Stream流等一系列新特性,學會了這些新特性,可以讓你實現高效編碼優雅編碼。1. 不受待見的空指針異常有個小故事:null引用最早是由英國科學家Tony Hoare提出的,多年後Hoare為自己的這個想法感到後悔莫及,並認為這是"價值百萬的重大失誤"。
  • 黑馬程式設計師java培訓機構如何?好口碑java培訓機構如何選?
    大大小小的Java培訓機構多如牛毛,幾個人經過簡單的包裝就會成為精英人士混淆視聽,因此想要知道黑馬程式設計師java培訓機構如何?好口碑java培訓機構如何選?我們就要考全面慮。想要知道一家java培訓機構好不好,品牌和實力是最為重要的,實力如何就要看師資力量了,擁有豐富實戰經驗的導師會結合學員過往的工作經歷、學習表現、階段性成績、項目成績等,為學員提供最佳的專業輔導。課程安排方面會凸顯這家java培訓機構的專業,畢竟專注的總歸是專業的,java培訓起來理論必須加實踐,因為最終的目的是打造符合企業需求的java人才。
  • Java中有關Null的9件事
    Java為什麼要保留null呢?null出現有一段時間了,並且我認為Java發明者知道null與它解決的問題相比帶來了更多的麻煩,但是null仍然陪伴著Java。我越發感到驚奇,因為java的設計原理是為了簡化事情,那就是為什麼沒有浪費時間在指針、操作符重載、多繼承實現的原因,null卻與此正好相反。
  • 面試重災區:Java的異常體系怎麼解?
    Java的異常體系結構Java異常體系的根類是 Throwable, 所以當寫在java代碼中寫throw拋出異常時,後面跟的對象必然是Throwable或其子類的對象。其中Exception異常是指一些可以恢復的異常, 例如常見的NullPointerException空指針異常。
  • 如何自學Java?避免誤區
    實際中在學習Java編程之前需要先學習一些編程基礎,新手零基礎怎麼自學java開發?自學java開發有什麼誤區?新手往往無非進行系統的學習,主要原因是自己搜集信息較為零散,學習過程中無專業的老師引導,自己會走進一些學習的誤區。
  • java是什麼文件格式?.java文件怎麼打開?
    Java是由Sun Microsystems公司於1995年5月推出的Java面向對象程序設計語言和Java平臺的總稱,是當今最流行的編程技術。  java是什麼文件?  Java文件是由Sun Microsystems公司於1995年5月推出的Java程序設計語言和Java平臺的總稱。
  • python和java哪個好
    我要如何選擇學習哪個語言,python和java哪個好就業?針對這個問題,下面我們一起來看一看。不過,這裡針對別人講的,有人跟你說Java比Python好或Python比Java好,聽到這種話,一定要根據自己的情況進行判斷,一般來講,用JAVA的就說JAVA好,用PYTHON的就說PYTHON好,因此徐老師建議,最好的肯定是最適合自己的。
  • Java基礎面試題簡單總結
    不可能指望程序能處理這樣的情況exception 表示一種設計或實現問題。java編譯器要求方法必須聲明拋出可能發生的非運行時異常,但是並不要求必須聲明拋出未被捕獲的運行時異常。另一種情況就是JAVA允許程式設計師擴展這種語義檢查,程式設計師可以創建自己的異常,並自由選擇在何時用throw關鍵字引發異常。所有的異常都是java.lang.Thowable的子類。43、垃圾回收的優點和原理。
  • Java面向對象程序設計之類的封裝
    所以為了避免程序中這種錯誤的情況的發生,在一般的開發中往往要將類中的屬性封裝(private),對範例TestPersonDemo2.java做了相應的修改後形成下面的程序TestPersonDemo3-1.java,如下所示:範例:TestPersonDemo3-1.java01 class Person02 {
  • 使用Java API在Jedis中實現DAO設計模式
    在本文中,我們將學習Jedis Java客戶端中的DAO設計模式和實現。DAO模式被實現為客戶端應用程式和資料庫之間的一層。客戶端應用程式不必依賴基礎資料庫交互API(低級)。Redis and Jedis在 Windows 上安裝 Redis在 Windows 上安裝 Eclipse創建 Maven 項目創建 SiteDaoRedisImplTest.java
  • 「007期」JavaSE面試題(七):異常
    (1)java.lang.NullPointerException 空指針異常;出現原因:調用了未經初始化的對象或者是不存在的對象。(3)java.lang.NumberFormatException 字符串轉換為數字異常;出現原因:字符型數據中包含非數字型字符。(4)java.lang.IndexOutOfBoundsException 數組角標越界異常,常見於操作數組對象時發生。
  • 如何在Windows10系統中配置java的JDK環境
    今天給大家分享一下如何在Windows10系統中配置java的JDK環境。下面驗證一下JDK的配置環境,打開cmd窗口界面,輸入 javac 命令,截圖如下:20.在cmd窗口輸入 java 命令,截圖如下:21.此時JDK環境配置成功了。
  • Java書架來啦!入門到進階必看書籍推薦
    Katty Sierra,Bert Bates 著 / 楊尊一 編譯 張然等 改編《Head First Java》是本完整的面向對象(object-oriented,OO)程序設計和Java的學習指導。此書是根據學習理論所設計的,讓你可以從學習程序語言的基礎開始一直到包括線程、網絡與分布式程序等項目。最重要的,你會學會如何像個面向對象開發者一樣去思考。
  • 一行JAVA代碼如何運行起來?
    JVM運行Java程序有兩種方式,分別是jar包和Class類文件,jar包是偏上層的方式,把所有程序都打包成一個jar包,便於交付測試人員測試、運維人員發布,它的運行邏輯是通過java.exe找到java自帶的GetMainClassName函數,該函數獲取JNIENV實例,並調用JarFileJNIENV實例中的GetMainfest()函數獲取MainClass函數,Main函數再調用Java.c
  • 如何應用策略設計模式分離JDBC資料庫連接中的外部環境信息
    軟體項目實訓及課程設計指導——如何應用策略設計模式分離JDBC資料庫連接中的外部環境信息1、什麼是策略(Strategy)設計模式策略設計模式把「算法」(也就是軟體應用系統中的業務規則或者待實現的功能等)和「環境」(封裝軟體應用系統在實際應用時的場景)相互分離
  • Lambda 表達式,簡潔優雅就是生產力!
    本文轉載自【微信公眾號:java進階架構師,ID:java_jiagoushi】經微信公眾號授權轉載,如需轉載與原文作者聯繫1什麼是Lambda?我們知道,對於一個Java變量,我們可以賦給其一個「值」。