如何優(yōu)雅的設(shè)計(jì)java異常

圖片發(fā)自簡書App

title: 如何優(yōu)雅的設(shè)計(jì)java異常
date: 2017-03-31 17:36:36
categories: Java
tags: 異常


本文轉(zhuǎn)載自Lrwin的java技術(shù)博客

導(dǎo)語

異常處理是程序開發(fā)中必不可少操作之一啄枕,但如何正確優(yōu)雅的對(duì)異常進(jìn)行處理確是一門學(xué)問孟抗,筆者根據(jù)自己的開發(fā)經(jīng)驗(yàn)來談一談我是如何對(duì)異常進(jìn)行處理的。
由于本文只作一些經(jīng)驗(yàn)之談,不涉及到基礎(chǔ)知識(shí)部分怨喘,如果讀者對(duì)異常的概念還很模糊翅敌,請先查看基礎(chǔ)知識(shí)。

如何選擇異常類型

異常的類別

正如我們所知道的幻碱,java中的異常的超類是java.lang.Throwable(后文省略為Throwable),它有兩個(gè)比較重要的子類,java.lang.Exception(后文省略為Exception)和java.lang.Error(后文省略為Error)绎狭,其中Error由JVM虛擬機(jī)進(jìn)行管理,如我們所熟知的OutOfMemoryError異常等,所以我們本文不關(guān)注Error異常褥傍,那么我們細(xì)說一下Exception異常儡嘶。
Exception異常有個(gè)比較重要的子類,叫做RuntimeException恍风。我們將RuntimeException或其他繼承自RuntimeException的子類稱為非受檢異常(unchecked Exception)社付,其他繼承自Exception異常的子類稱為受檢異常(checked Exception)。本文重點(diǎn)來關(guān)注一下受檢異常和非受檢異常這兩種異常邻耕。

如何選擇異常

從筆者的開發(fā)經(jīng)驗(yàn)來看鸥咖,如果在一個(gè)應(yīng)用中,需要開發(fā)一個(gè)方法(如某個(gè)功能的service方法)兄世,這個(gè)方法如果中間可能出現(xiàn)異常啼辣,那么你需要考慮這個(gè)異常出現(xiàn)之后是否調(diào)用者可以處理,并且你是否希望調(diào)用者進(jìn)行處理御滩,如果調(diào)用者可以處理鸥拧,并且你也希望調(diào)用者進(jìn)行處理,那么就要拋出受檢異常削解,提醒調(diào)用者在使用你的方法時(shí)富弦,考慮到如果拋出異常時(shí)如果進(jìn)行處理,相似的氛驮,如果在寫某個(gè)方法時(shí)腕柜,你認(rèn)為這是個(gè)偶然異常,理論上說矫废,你覺得運(yùn)行時(shí)可能會(huì)碰到什么問題盏缤,而這些問題也許不是必然發(fā)生的,也不需要調(diào)用者顯示的通過異常來判斷業(yè)務(wù)流程操作的蓖扑,那么這時(shí)就可以使用一個(gè)RuntimeException這樣的非受檢異常.
好了唉铜,估計(jì)我上邊說的這段話,你讀了很多遍也依然覺得晦澀了律杠。
那么潭流,請跟著我的思路,在慢慢領(lǐng)會(huì)一下柜去。

什么時(shí)候才需要拋異常

首先我們需要了解一個(gè)問題灰嫉,什么時(shí)候才需要拋異常?異常的設(shè)計(jì)是方便給開發(fā)者使用的诡蜓,但不是亂用的熬甫,筆者對(duì)于什么時(shí)候拋異常這個(gè)問題也問了很多朋友,能給出準(zhǔn)確答案的確實(shí)不多。其實(shí)這個(gè)問題很簡單椿肩,如果你覺得某些”問題”解決不了了瞻颂,那么你就可以拋出異常了。比如郑象,你在寫一個(gè)service,其中在寫到某段代碼處,你發(fā)現(xiàn)可能會(huì)產(chǎn)生問題贡这,那么就請拋出異常吧,相信我厂榛,你此時(shí)拋出異常將是一個(gè)最佳時(shí)機(jī)盖矫。

應(yīng)該拋出怎樣的異常

了解完了什么時(shí)候才需要拋出異常后,我們再思考一個(gè)問題击奶,真的當(dāng)我們拋出異常時(shí)辈双,我們應(yīng)該選用怎樣的異常呢?究竟是受檢異常還是非受檢異常呢(RuntimeException)呢柜砾?我來舉例說明一下這個(gè)問題湃望,先從受檢異常說起,比如說有這樣一個(gè)業(yè)務(wù)邏輯,需要從某文件中讀取某個(gè)數(shù)據(jù)痰驱,這個(gè)讀取操作可能是由于文件被刪除等其他問題導(dǎo)致無法獲取從而出現(xiàn)讀取錯(cuò)誤证芭,那么就要從redis或mysql數(shù)據(jù)庫中再去獲取此數(shù)據(jù),參考如下代碼,getKey(Integer)為入口程序.

public String getKey(Integer key){
    String  value;
    try {
        InputStream inputStream = getFiles("/file/nofile");
        //接下來從流中讀取key的value指
        value = ...;
    } catch (Exception e) {
        //如果拋出異常將從mysql或者redis進(jìn)行取之
        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讀取錯(cuò)誤",e.getCause());
    }
    return inputStream;
}

ok担映,看了以上代碼以后废士,你也許心中有一些想法,原來受檢異秤辏可以控制義務(wù)邏輯官硝,對(duì),沒錯(cuò)四敞,通過受檢異常真的可以控制業(yè)務(wù)邏輯泛源,但是切記不要這樣使用,我們應(yīng)該合理的拋出異常忿危,因?yàn)槌绦虮旧聿攀橇鞒蹋惓5淖饔脙H僅是當(dāng)你進(jìn)行不下去的時(shí)候找到的一個(gè)借口而已没龙,它并不能當(dāng)成控制程序流程的入口或出口铺厨,如果這樣使用的話,是在將異常的作用擴(kuò)大化硬纤,這樣將會(huì)導(dǎo)致代碼復(fù)雜程度的增加解滓,耦合性會(huì)提高,代碼可讀性降低等問題筝家。那么就一定不要使用這樣的異常嗎洼裤?其實(shí)也不是,在真的有這樣的需求的時(shí)候溪王,我們可以這樣使用腮鞍,只是切記值骇,不要把它真的當(dāng)成控制流程的工具或手段。那么究竟什么時(shí)候才要拋出這樣的異常呢移国?要考慮吱瘩,如果調(diào)用者調(diào)用出錯(cuò)后,一定要讓調(diào)用者對(duì)此錯(cuò)誤進(jìn)行處理才可以迹缀,滿足這樣的要求時(shí)使碾,我們才會(huì)考慮使用受檢異常。
接下來祝懂,我們來看一下非受檢異常呢(RuntimeException)票摇,對(duì)于RuntimeException這種異常,我們其實(shí)很多見砚蓬,比如java.lang.NullPointerException/java.lang.IllegalArgumentException等兄朋,那么這種異常我們時(shí)候拋出呢?當(dāng)我們在寫某個(gè)方法的時(shí)候怜械,可能會(huì)偶然遇到某個(gè)錯(cuò)誤颅和,我們認(rèn)為這個(gè)問題時(shí)運(yùn)行時(shí)可能為發(fā)生的,并且理論上講缕允,沒有這個(gè)問題的話峡扩,程序?qū)?huì)正常執(zhí)行的時(shí)候,它不強(qiáng)制要求調(diào)用者一定要捕獲這個(gè)異常障本,此時(shí)拋出RuntimeException異常,舉個(gè)例子教届,當(dāng)傳來一個(gè)路徑的時(shí)候,需要返回一個(gè)路徑對(duì)應(yīng)的File對(duì)象:

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;
}

上述例子表明驾霜,如果調(diào)用者調(diào)用getFiles(String)的時(shí)候如果path是空案训,那么就拋出空指針異常(它是RuntimeException的子類),調(diào)用者不用顯示的進(jìn)行try…catch…操作進(jìn)行強(qiáng)制處理.這就要求調(diào)用者在調(diào)用這樣的方法時(shí)先進(jìn)行驗(yàn)證,避免發(fā)生RuntimeException.如下:

public void test() {
    String path = "/a/b.png";
    if(null != path && !"".equals(path)){
        myTest.getFiles("");
    }
}

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

    return file;
}

應(yīng)該選用哪種異常

通過以上的描述和舉例粪糙,可以總結(jié)出一個(gè)結(jié)論强霎,RuntimeException異常和受檢異常之間的區(qū)別就是:是否強(qiáng)制要求調(diào)用者必須處理此異常,如果強(qiáng)制要求調(diào)用者必須進(jìn)行處理蓉冈,那么就使用受檢異常城舞,否則就選擇非受檢異常(RuntimeException)。一般來講寞酿,如果沒有特殊的要求家夺,我們建議使用RuntimeException異常。

場景介紹和技術(shù)選型

架構(gòu)描述

正如我們所知伐弹,傳統(tǒng)的項(xiàng)目都是以MVC框架為基礎(chǔ)進(jìn)行開發(fā)的拉馋,本文主要從使用restful風(fēng)格接口的設(shè)計(jì)來體驗(yàn)一下異常處理的優(yōu)雅。
我們把關(guān)注點(diǎn)放在restful的api層(和web中的controller層類似)和service層,研究一下在service中如何拋出異常煌茴,然后api層如何進(jìn)行捕獲并且轉(zhuǎn)化異常随闺。
使用的技術(shù)是:spring-boot,jpa(hibernate),mysql,如果對(duì)這些技術(shù)不是太熟悉,讀者需要自行閱讀相關(guān)材料景馁。

業(yè)務(wù)場景描述

選擇一個(gè)比較簡單的業(yè)務(wù)場景板壮,以電商中的收貨地址管理為例,用戶在移動(dòng)端進(jìn)行購買商品時(shí)合住,需要進(jìn)行收貨地址管理绰精,在項(xiàng)目中,提供一些給移動(dòng)端進(jìn)行訪問的api接口透葛,如:添加收貨地址笨使,刪除收貨地址,更改收貨地址僚害,默認(rèn)收貨地址設(shè)置硫椰,收貨地址列表查詢,單個(gè)收貨地址查詢等接口萨蚕。

構(gòu)建約束條件

ok靶草,這個(gè)是設(shè)置好的一個(gè)很基本的業(yè)務(wù)場景,當(dāng)然岳遥,無論什么樣的api操作奕翔,其中都包含一些規(guī)則:

  • 添加收貨地址:
    入?yún)?

    1. 用戶id

    2. 收貨地址實(shí)體信息


      約束:

    3. 用戶id不能為空,且此用戶確實(shí)是存在 的

    4. 收貨地址的必要字段不能為 空

    5. 如果用戶還沒有收貨地址浩蓉,當(dāng)此收貨地址創(chuàng)建時(shí)設(shè)置成默認(rèn)收貨地址 —


  • 刪除收貨地址:
    入?yún)?

    1. 用戶id
    2. 收貨地址id

    約束:

    1. 用戶id不能為空派继,且此用戶確實(shí)是存在的

    2. 收貨地址不能為空,且此收貨地址確實(shí)是存在的

      1. 判斷此收貨地址是否是用戶的收貨地址
      2. 判斷此收貨地址是否為默認(rèn)收貨地址捻艳,如果是默認(rèn)收貨地址驾窟,那么不能進(jìn)行刪除

  • 更改收貨地址:
    入?yún)?

    1. 用戶id

      1. 收貨地址id

      約束:

      1. 用戶id不能為空,且此用戶確實(shí)是存在的
    2. 收貨地址不能為空认轨,且此收貨地址確實(shí)是存在的

    3. 判斷此收貨地址是否是用戶的收貨地址


  • 默認(rèn)地址設(shè)置:
    入?yún)?

    1. 用戶id
    2. 收貨地址id

    約束:

    1. 用戶id不能為空绅络,且此用戶確實(shí)是存在的
      1. 收貨地址不能為空,且此收貨地址確實(shí)是存在的
      2. 判斷此收貨地址是否是用戶的收貨地址

  • 收貨地址列表查詢:
    入?yún)?

    1. 用戶id


      約束:

    2. 用戶id不能為空好渠,且此用戶確實(shí)是存在的


  • 單個(gè)收貨地址查詢:
    入?yún)?

    1. 用戶id

      1. 收貨地址id

      約束:

      1. 用戶id不能為空昨稼,且此用戶確實(shí)是存在的
    2. 收貨地址不能為空,且此收貨地址確實(shí)是存在的

    3. 判斷此收貨地址是否是用戶的收貨地址


約束判斷和技術(shù)選型

對(duì)于上述列出的約束條件和功能列表拳锚,我選擇幾個(gè)比較典型的異常處理場景進(jìn)行分析:添加收貨地址,刪除收貨地址寻行,獲取收貨地址列表霍掺。
那么應(yīng)該有哪些必要的知識(shí)儲(chǔ)備呢,讓我們看一下收貨地址這個(gè)功能:
添加收貨地址中需要對(duì)用戶id和收貨地址實(shí)體信息就行校驗(yàn),那么對(duì)于非空的判斷杆烁,我們?nèi)绾芜M(jìn)行工具的選擇呢牙丽?傳統(tǒng)的判斷如下:

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

上邊的例子,如果只判斷uid為空還好兔魂,如果再去判斷address這個(gè)實(shí)體中的某些必要屬性是否為空烤芦,在字段很多的情況下,這無非是災(zāi)難性的析校。
那我們應(yīng)該怎么進(jìn)行這些入?yún)⒌呐袛嗄毓孤蓿o大家介紹兩個(gè)知識(shí)點(diǎn):

  1. Guava中的Preconditions類實(shí)現(xiàn)了很多入?yún)⒎椒ǖ呐袛?/li>
  2. jsr 303的validation規(guī)范(目前實(shí)現(xiàn)比較全的是hibernate實(shí)現(xiàn)的hibernate-validator)
    如果使用了這兩種推薦技術(shù),那么入?yún)⒌呐袛鄷?huì)變得簡單很多智玻。推薦大家多使用這些成熟的技術(shù)和jar工具包遂唧,他可以減少很多不必要的工作量。我們只需要把重心放到業(yè)務(wù)邏輯上吊奢。而不會(huì)因?yàn)檫@些入?yún)⒌呐袛嗟⒄`更多的時(shí)間盖彭。

如何優(yōu)雅的設(shè)計(jì)java異常

domain介紹

根據(jù)項(xiàng)目場景來看,需要兩個(gè)domain模型页滚,一個(gè)是用戶實(shí)體召边,一個(gè)是地址實(shí)體.
Address domain如下:

@Entity
@Data
public class Address {
    @Id
    @GeneratedValue
    private Integer id;
    private String province;//省
    private String city;//市
    private String county;//區(qū)
    private Boolean isDefault;//是否是默認(rèn)地址

    @ManyToOne(cascade={CascadeType.ALL})
    @JoinColumn(name="uid")
    private User user;
}

User domain如下:

@Entity
@Data
public class User {
    @Id
   @GeneratedValue
   private Integer id;
   private String name;//姓名

    @OneToMany(cascade= CascadeType.ALL,mappedBy="user",fetch = FetchType.LAZY)
        private Set<Address> addresses;
}

ok,上邊是一個(gè)模型關(guān)系,用戶-收貨地址的關(guān)系是1-n的關(guān)系裹驰。上邊的@Data是使用了一個(gè)叫做lombok的工具隧熙,它自動(dòng)生成了Setter和Getter等方法,用起來非常方便邦马,感興趣的讀者可以自行了解一下贱鼻。

dao介紹

數(shù)據(jù)連接層,我們使用了spring-data-jpa這個(gè)框架滋将,它要求我們只需要繼承框架提供的接口邻悬,并且按照約定對(duì)方法進(jìn)行取名,就可以完成我們想要的數(shù)據(jù)庫操作随闽。
用戶數(shù)據(jù)庫操作如下:

@Repository
public interface IUserDao extends JpaRepository<User,Integer> {

}

收貨地址操作如下:

@Repository
public interface IAddressDao extends JpaRepository<Address,Integer> {

}

正如讀者所看到的父丰,我們的DAO只需要繼承JpaRepository,它就已經(jīng)幫我們完成了基本的CURD等操作,如果想了解更多關(guān)于spring-data的這個(gè)項(xiàng)目掘宪,請參考一下spring的官方文檔蛾扇,它比不方案我們對(duì)異常的研究。

Service異常設(shè)計(jì)

ok魏滚,終于到了我們的重點(diǎn)了镀首,我們要完成service一些的部分操作:添加收貨地址,刪除收貨地址鼠次,獲取收貨地址列表.
首先看我的service接口定義:

public interface IAddressService {

/**
 * 創(chuàng)建收貨地址
 * @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);
}

我們來關(guān)注一下實(shí)現(xiàn):

添加收貨地址

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

入?yún)?

  1. 用戶id
  2. 收貨地址實(shí)體信息

約束:

  1. 用戶id不能為空更哄,且此用戶確實(shí)是存在的
  2. 收貨地址的必要字段不能為空
  3. 如果用戶還沒有收貨地址芋齿,當(dāng)此收貨地址創(chuàng)建時(shí)設(shè)置成默認(rèn)收貨地址

先看以下代碼實(shí)現(xiàn):

 @Override
public Address createAddress(Integer uid, Address address) {
    //============ 以下為約束條件   ==============
    //1.用戶id不能為空,且此用戶確實(shí)是存在的
    Preconditions.checkNotNull(uid);
    User user = userDao.findOne(uid);
    if(null == user){
        throw new RuntimeException("找不到當(dāng)前用戶!");
    }
    //2.收貨地址的必要字段不能為空
    BeanValidators.validateWithException(validator, address);
    //3.如果用戶還沒有收貨地址成翩,當(dāng)此收貨地址創(chuàng)建時(shí)設(shè)置成默認(rèn)收貨地址
    if(ObjectUtils.isEmpty(user.getAddresses())){
        address.setIsDefault(true);
    }

    //============ 以下為正常執(zhí)行的業(yè)務(wù)邏輯   ==============
    address.setUser(user);
    Address result = addressDao.save(address);
    return result;
}

其中觅捆,已經(jīng)完成了上述所描述的三點(diǎn)約束條件,當(dāng)三點(diǎn)約束條件都滿足時(shí)麻敌,才可以進(jìn)行正常的業(yè)務(wù)邏輯栅炒,否則將拋出異常(一般在此處建議拋出運(yùn)行時(shí)異常-RuntimeException)。

介紹以下以上我所用到的技術(shù):

  1. Preconfitions.checkNotNull(T t)這個(gè)是使用Guava中的com.google.common.base.Preconditions進(jìn)行判斷的术羔,因?yàn)閟ervice中用到的驗(yàn)證較多赢赊,所以建議將Preconfitions改成靜態(tài)導(dǎo)入的方式:

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

    當(dāng)然Guava的github中的說明也建議我們這樣使用。

  2. BeanValidators.validateWithException(validator, address);
    這個(gè)使用了hibernate實(shí)現(xiàn)的jsr 303規(guī)范來做的聂示,需要傳入一個(gè)validator和一個(gè)需要驗(yàn)證的實(shí)體,那么validator是如何獲取的呢,如下:

    @Configuration
    public class BeanConfigs {

    @Bean
    public javax.validation.Validator getValidator(){
        return new LocalValidatorFactoryBean();
    }
    

    }

他將獲取一個(gè)Validator對(duì)象域携,然后我們在service中進(jìn)行注入便可以使用了:

 @Autowired     
private Validator validator ;

那么BeanValidators這個(gè)類是如何實(shí)現(xiàn)的?其實(shí)實(shí)現(xiàn)方式很簡單鱼喉,只要去判斷jsr 303的標(biāo)注注解就ok了秀鞭。
那么jsr 303的注解寫在哪里了呢?當(dāng)然是寫在address實(shí)體類中了:

@Entity
@Setter
@Getter
public class Address {
@Id
    @GeneratedValue
    private Integer id;
    @NotNull
private String province;//省
@NotNull
private String city;//市
@NotNull
private String county;//區(qū)
private Boolean isDefault = false;//是否是默認(rèn)地址

@ManyToOne(cascade={CascadeType.ALL})
@JoinColumn(name="uid")
private User user;
}

寫好你需要的約束條件來進(jìn)行判斷扛禽,如果合理的話锋边,才可以進(jìn)行業(yè)務(wù)操作,從而對(duì)數(shù)據(jù)庫進(jìn)行操作编曼。
這塊的驗(yàn)證是必須的豆巨,一個(gè)最主要的原因是:這樣的驗(yàn)證可以避免臟數(shù)據(jù)的插入。如果讀者有正式上線的經(jīng)驗(yàn)的話掐场,就可以理解這樣的一個(gè)事情往扔,任何的代碼錯(cuò)誤都可以容忍和修改,但是如果出現(xiàn)了臟數(shù)據(jù)問題熊户,那么它有可能是一個(gè)毀滅性的災(zāi)難萍膛。程序的問題可以修改,但是臟數(shù)據(jù)的出現(xiàn)有可能無法恢復(fù)嚷堡。所以這就是為什么在service中一定要判斷好約束條件蝗罗,再進(jìn)行業(yè)務(wù)邏輯操作的原因了。

  1. 此處的判斷為業(yè)務(wù)邏輯判斷蝌戒,是從業(yè)務(wù)角度來進(jìn)行篩選判斷的串塑,除此之外,有可能在很多場景中都會(huì)有不同的業(yè)務(wù)條件約束北苟,只需要按照要求來做就好桩匪。

對(duì)于約束條件的總結(jié)如下:

  • 基本判斷約束(null值等基本判斷)
  • 實(shí)體屬性約束(滿足jsr 303等基礎(chǔ)判斷)
  • 業(yè)務(wù)條件約束(需求提出的不同的業(yè)務(wù)約束)

當(dāng)這個(gè)三點(diǎn)都滿足時(shí),才可以進(jìn)行下一步操作

ok,基本介紹了如何做一個(gè)基礎(chǔ)的判斷友鼻,那么再回到異常的設(shè)計(jì)問題上吸祟,上述代碼已經(jīng)很清楚的描述如何在適當(dāng)?shù)奈恢煤侠淼呐袛嘁粋€(gè)異常了瑟慈,那么如何合理的拋出異常呢桃移?
只拋出RuntimeException就算是優(yōu)雅的拋出異常嗎屋匕?當(dāng)然不是,對(duì)于service中的拋出異常借杰,筆者認(rèn)為大致有兩種拋出的方法:

  1. 拋出帶狀態(tài)碼RumtimeException異常
  2. 拋出指定類型的RuntimeException異常

相對(duì)這兩種異常的方式進(jìn)行結(jié)束过吻,第一種異常指的是我所有的異常都拋RuntimeException異常,但是需要帶一個(gè)狀態(tài)碼蔗衡,調(diào)用者可以根據(jù)狀態(tài)碼再去查詢究竟service拋出了一個(gè)什么樣的異常纤虽。
第二種異常是指在service中拋出什么樣的異常就自定義一個(gè)指定的異常錯(cuò)誤,然后在進(jìn)行拋出異常绞惦。
一般來講逼纸,如果系統(tǒng)沒有別的特殊需求的時(shí)候,在開發(fā)設(shè)計(jì)中济蝉,建議使用第二種方式杰刽。但是比如說像基礎(chǔ)判斷的異常,就可以完全使用guava給我們提供的類庫進(jìn)行操作王滤。jsr 303異常也可以使用自己封裝好的異常判斷類進(jìn)行操作贺嫂,因?yàn)檫@兩種異常都是屬于基礎(chǔ)判斷,不需要為它們指定特殊的異常雁乡。但是對(duì)于第三點(diǎn)義務(wù)條件約束判斷拋出的異常第喳,就需要拋出指定類型的異常了。
對(duì)于

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

定義一個(gè)特定的異常類來進(jìn)行這個(gè)義務(wù)異常的判斷:

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

public NotFindUserException(String message) {
    super(message);
}
}

然后將此處改為:

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

or

throw new NotFindUserException();

ok,通過以上對(duì)service層的修改踱稍,代碼更改如下:

@Override
public Address createAddress(Integer uid, Address address) {
    //============ 以下為約束條件   ==============
    //1.用戶id不能為空曲饱,且此用戶確實(shí)是存在的
    checkNotNull(uid);
    User user = userDao.findOne(uid);
    if(null == user){
        throw new NotFindUserException("找不到當(dāng)前用戶!");
    }
    //2.收貨地址的必要字段不能為空
    BeanValidators.validateWithException(validator, address);
    //3.如果用戶還沒有收貨地址,當(dāng)此收貨地址創(chuàng)建時(shí)設(shè)置成默認(rèn)收貨地址
    if(ObjectUtils.isEmpty(user.getAddresses())){
        address.setIsDefault(true);
    }

    //============ 以下為正常執(zhí)行的業(yè)務(wù)邏輯   ==============
    address.setUser(user);
    Address result = addressDao.save(address);
    return result;
}

這樣的service就看起來穩(wěn)定性和理解性就比較強(qiáng)了珠月。

刪除收貨地址:

入?yún)?

  1. 用戶id
  2. 收貨地址id

約束:

  1. 用戶id不能為空扩淀,且此用戶確實(shí)是存在的
  2. 收貨地址不能為空,且此收貨地址確實(shí)是存在的
  3. 判斷此收貨地址是否是用戶的收貨地址
  4. 判斷此收貨地址是否為默認(rèn)收貨地址桥温,如果是默認(rèn)收貨地址引矩,那么不能進(jìn)行刪除

它與上述添加收貨地址類似,故不再贅述侵浸,delete的service設(shè)計(jì)如下:

@Override
public void deleteAddress(Integer uid, Integer aid) {
    //============ 以下為約束條件   ==============
    //1.用戶id不能為空旺韭,且此用戶確實(shí)是存在的
    checkNotNull(uid);
    User user = userDao.findOne(uid);
    if(null == user){
        throw new NotFindUserException();
    }
    //2.收貨地址不能為空,且此收貨地址確實(shí)是存在的
    checkNotNull(aid);
    Address address = addressDao.findOne(aid);
    if(null == address){
        throw new NotFindAddressException();
    }
    //3.判斷此收貨地址是否是用戶的收貨地址
    if(!address.getUser().equals(user)){
        throw new NotMatchUserAddressException();
    }
    //4.判斷此收貨地址是否為默認(rèn)收貨地址掏觉,如果是默認(rèn)收貨地址区端,那么不能進(jìn)行刪除
    if(address.getIsDefault()){
       throw  new DefaultAddressNotDeleteException();
    }

    //============ 以下為正常執(zhí)行的業(yè)務(wù)邏輯   ==============
    addressDao.delete(address);
}

設(shè)計(jì)了相關(guān)的四個(gè)異常類:
NotFindUserException,NotFindAddressException,NotMatchUserAddressException,DefaultAddressNotDeleteException.根據(jù)不同的業(yè)務(wù)需求拋出不同的異常。

獲取收貨地址列表:

入?yún)?

  1. 用戶id

約束:

  1. 用戶id不能為空澳腹,且此用戶確實(shí)是存在的

代碼如下:

 @Override
public List<Address> listAddresses(Integer uid) {
    //============ 以下為約束條件   ==============
    //1.用戶id不能為空织盼,且此用戶確實(shí)是存在的
    checkNotNull(uid);
    User user = userDao.findOne(uid);
    if(null == user){
        throw new NotFindUserException();
    }

    //============ 以下為正常執(zhí)行的業(yè)務(wù)邏輯   ==============
    User result = userDao.findOne(uid);
    return result.getAddresses();
}

api異常設(shè)計(jì)

大致有兩種拋出的方法:

  1. 拋出帶狀態(tài)碼RumtimeException異常
  2. 拋出指定類型的RuntimeException異常

這個(gè)是在設(shè)計(jì)service層異常時(shí)提到的杨何,通過對(duì)service層的介紹,我們在service層拋出異常時(shí)選擇了第二種拋出的方式沥邻,不同的是危虱,在api層拋出異常我們需要使用這兩種方式進(jìn)行拋出:要指定api異常的類型,并且要指定相關(guān)的狀態(tài)碼唐全,然后才將異常拋出埃跷,這種異常設(shè)計(jì)的核心是讓調(diào)用api的使用者更能清楚的了解發(fā)生異常的詳細(xì)信息,除了拋出異常外邮利,我們還需要將狀態(tài)碼對(duì)應(yīng)的異常詳細(xì)信息以及異常有可能發(fā)生的問題制作成一個(gè)對(duì)應(yīng)的表展示給用戶弥雹,方便用戶的查詢。(如github提供的api文檔延届,微信提供的api文檔等),還有一個(gè)好處:如果用戶需要自定義提示消息剪勿,可以根據(jù)返回的狀態(tài)碼進(jìn)行提示的修改。

api驗(yàn)證約束

首先對(duì)于api的設(shè)計(jì)來說方庭,需要存在一個(gè)dto對(duì)象厕吉,這個(gè)對(duì)象負(fù)責(zé)和調(diào)用者進(jìn)行數(shù)據(jù)的溝通和傳遞,然后dto->domain在傳給service進(jìn)行操作二鳄,這一點(diǎn)一定要注意赴涵,第二點(diǎn),除了說道的service需要進(jìn)行基礎(chǔ)判斷(null判斷)和jsr 303驗(yàn)證以外订讼,同樣的髓窜,api層也需要進(jìn)行相關(guān)的驗(yàn)證,如果驗(yàn)證不通過的話欺殿,直接返回給調(diào)用者寄纵,告知調(diào)用失敗,不應(yīng)該帶著不合法的數(shù)據(jù)再進(jìn)行對(duì)service的訪問脖苏,那么讀者可能會(huì)有些迷惑程拭,不是service已經(jīng)進(jìn)行驗(yàn)證了,為什么api層還需要進(jìn)行驗(yàn)證么棍潘?這里便設(shè)計(jì)到了一個(gè)概念:編程中的墨菲定律恃鞋,如果api層的數(shù)據(jù)驗(yàn)證疏忽了,那么有可能不合法數(shù)據(jù)就帶到了service層亦歉,進(jìn)而講臟數(shù)據(jù)保存到了數(shù)據(jù)庫恤浪。

所以縝密編程的核心是:永遠(yuǎn)不要相信收到的數(shù)據(jù)是合法的。

api異常設(shè)計(jì)

設(shè)計(jì)api層異常時(shí)肴楷,正如我們上邊所說的水由,需要提供錯(cuò)誤碼和錯(cuò)誤信息,那么可以這樣設(shè)計(jì)赛蔫,提供一個(gè)通用的api超類異常砂客,其他不同的api異常都繼承自這個(gè)超類:

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.以默認(rèn)地址不能刪除為例:

public class ApiDefaultAddressNotDeleteException extends ApiException {

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

}

AddressErrorCode.DefaultAddressNotDeleteErrorCode就是需要提供給調(diào)用者的錯(cuò)誤碼泥张。錯(cuò)誤碼類如下:

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

ok,那么api層的異常就已經(jīng)設(shè)計(jì)完了,在此多說一句鞠值,AddressErrorCode錯(cuò)誤碼類存放了可能出現(xiàn)的錯(cuò)誤碼媚创,更合理的做法是把他放到配置文件中進(jìn)行管理。

api處理異常

api層會(huì)調(diào)用service層齿诉,然后來處理service中出現(xiàn)的所有異常筝野,首先,需要保證一點(diǎn)粤剧,一定要讓api層非常輕,基本上做成一個(gè)轉(zhuǎn)發(fā)的功能就好(接口參數(shù)挥唠,傳遞給service參數(shù)抵恋,返回給調(diào)用者數(shù)據(jù),這三個(gè)基本功能),然后就要在傳遞給service參數(shù)的那個(gè)方法調(diào)用上進(jìn)行異常處理宝磨。

此處僅以添加地址為例:

 @Autowired
private 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){//未知錯(cuò)誤
        throw new ApiException(e);
    }
    AddressDTO resultDTO = new AddressDTO();
    BeanUtils.copyProperties(result,resultDTO);
    resultDTO.setUid(result.getUser().getId());

    return resultDTO;
}

這里的處理方案是調(diào)用service時(shí)弧关,判斷異常的類型,然后將任何service異常都轉(zhuǎn)化成api異常唤锉,然后拋出api異常世囊,這是常用的一種異常轉(zhuǎn)化方式。相似刪除收貨地址和獲取收貨地址也類似這樣處理窿祥,在此株憾,不在贅述。

api異常轉(zhuǎn)化

已經(jīng)講解了如何拋出異常和何如將service異常轉(zhuǎn)化為api異常晒衩,那么轉(zhuǎn)化成api異常直接拋出是否就完成了異常處理呢嗤瞎? 答案是否定的,當(dāng)拋出api異常后听系,我們需要把a(bǔ)pi異常返回的數(shù)據(jù)(json or xml)讓用戶看懂贝奇,那么需要把a(bǔ)pi異常轉(zhuǎn)化成dto對(duì)象(ErrorDTO),看如下代碼:

@ControllerAdvice(annotations = RestController.class)

class ApiExceptionHandlerAdvice {

/**
 * Handle exceptions thrown by handlers.
 */
@ExceptionHandler(value = Exception.class)
@ResponseBody
public 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
@Getter
class ErrorDTO{
    private Long errorCode;
    private String tip;
}

}

ok,這樣就完成了api異常轉(zhuǎn)化成用戶可以讀懂的DTO對(duì)象了,代碼中用到了@ControllerAdvice靠胜,這是spring MVC提供的一個(gè)特殊的切面處理掉瞳。

當(dāng)調(diào)用api接口發(fā)生異常時(shí),用戶也可以收到正常的數(shù)據(jù)格式了,比如當(dāng)沒有用戶(uid為2)時(shí)浪漠,卻為這個(gè)用戶添加收貨地址,postman(Google plugin 用于模擬http請求)之后的數(shù)據(jù):

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

總結(jié)

本文只從如何設(shè)計(jì)異常作為重點(diǎn)來講解陕习,涉及到的api傳輸和service的處理,還有待優(yōu)化郑藏,比如api接口訪問需要使用https進(jìn)行加密衡查,api接口需要OAuth2.0授權(quán)或api接口需要簽名認(rèn)證等問題,文中都未曾提到必盖,本文的重心在于異常如何處理拌牲,所以讀者只需關(guān)注涉及到異常相關(guān)的問題和處理方式就可以了俱饿。希望本篇文章對(duì)你理解異常有所幫助。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末塌忽,一起剝皮案震驚了整個(gè)濱河市拍埠,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌土居,老刑警劉巖枣购,帶你破解...
    沈念sama閱讀 216,324評(píng)論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異擦耀,居然都是意外死亡棉圈,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,356評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門眷蜓,熙熙樓的掌柜王于貴愁眉苦臉地迎上來分瘾,“玉大人,你說我怎么就攤上這事吁系〉抡伲” “怎么了?”我有些...
    開封第一講書人閱讀 162,328評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵汽纤,是天一觀的道長上岗。 經(jīng)常有香客問我,道長蕴坪,這世上最難降的妖魔是什么肴掷? 我笑而不...
    開封第一講書人閱讀 58,147評(píng)論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮辞嗡,結(jié)果婚禮上捆等,老公的妹妹穿的比我還像新娘。我一直安慰自己续室,他們只是感情好栋烤,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,160評(píng)論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著挺狰,像睡著了一般明郭。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上丰泊,一...
    開封第一講書人閱讀 51,115評(píng)論 1 296
  • 那天薯定,我揣著相機(jī)與錄音,去河邊找鬼瞳购。 笑死话侄,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播年堆,決...
    沈念sama閱讀 40,025評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼吞杭,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了变丧?” 一聲冷哼從身側(cè)響起芽狗,我...
    開封第一講書人閱讀 38,867評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎痒蓬,沒想到半個(gè)月后童擎,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,307評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡攻晒,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,528評(píng)論 2 332
  • 正文 我和宋清朗相戀三年顾复,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片炎辨。...
    茶點(diǎn)故事閱讀 39,688評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡捕透,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出碴萧,到底是詐尸還是另有隱情,我是刑警寧澤末购,帶...
    沈念sama閱讀 35,409評(píng)論 5 343
  • 正文 年R本政府宣布破喻,位于F島的核電站,受9級(jí)特大地震影響盟榴,放射性物質(zhì)發(fā)生泄漏曹质。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,001評(píng)論 3 325
  • 文/蒙蒙 一擎场、第九天 我趴在偏房一處隱蔽的房頂上張望羽德。 院中可真熱鬧,春花似錦迅办、人聲如沸宅静。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,657評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽姨夹。三九已至,卻和暖如春矾策,著一層夾襖步出監(jiān)牢的瞬間磷账,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,811評(píng)論 1 268
  • 我被黑心中介騙來泰國打工贾虽, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留逃糟,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,685評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像绰咽,于是被迫代替她去往敵國和親菇肃。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,573評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法剃诅,類相關(guān)的語法巷送,內(nèi)部類的語法,繼承相關(guān)的語法矛辕,異常的語法笑跛,線程的語...
    子非魚_t_閱讀 31,622評(píng)論 18 399
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,070評(píng)論 25 707
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)聊品,斷路器飞蹂,智...
    卡卡羅2017閱讀 134,651評(píng)論 18 139
  • 一. Java基礎(chǔ)部分.................................................
    wy_sure閱讀 3,810評(píng)論 0 11
  • 早上在必勝客吃完早飯,去人潮涌動(dòng)的新開業(yè)的盒馬鮮生逛了一圈回來翻屈,坐在沙發(fā)上抱著手機(jī)一看陈哑,就有點(diǎn)犯困不想起來了。想到...
    聰穎不雷人閱讀 709評(píng)論 3 3