0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
电子发烧友
开通电子发烧友VIP会员 尊享10大特权
海量资料免费下载
精品直播免费看
优质内容免费畅学
课程9折专享价
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Service層的異常處理

jf_ro2CN3Fa ? 來(lái)源:JavaEdge ? 2024-01-08 11:29 ? 次閱讀

來(lái)源:JavaEdge

0 前言

一般初學(xué)者學(xué)習(xí)編碼和[錯(cuò)誤處理]時(shí),先知道[編程語(yǔ)言]有一種處理錯(cuò)誤的形式或約定(如Java就拋異常),然后就開(kāi)始用這些工具。但卻忽視這問(wèn)題本質(zhì):「處理錯(cuò)誤是為了寫(xiě)正確程序」 ??墒?/p>

1 啥叫“正確”?

「由解決的問(wèn)題決定的。問(wèn)題不同,解決方案不同?!?/strong>

如一個(gè)web接口接受用戶請(qǐng)求,參數(shù)age,也許業(yè)務(wù)要求字段是0~150之間整數(shù)。如輸入字符串或負(fù)數(shù)就肯定不接受。一般在后端某地做輸入合法性檢查,不過(guò)就拋異常。

但歸根到底這問(wèn)題“正確”解決方法總是要以某種形式提示用戶。而提示用戶是某種前端工作,就要看界面是app,H5+AJAX還是類似于[jsp]的服務(wù)器產(chǎn)生界面。不管啥,你要根據(jù)需求去”設(shè)計(jì)一個(gè)修復(fù)錯(cuò)誤“的流程。如一個(gè)常見(jiàn)的流程要后端拋異常,然后一路到某個(gè)集中處理錯(cuò)誤的代碼,將其轉(zhuǎn)換為某個(gè)HTTP的錯(cuò)誤(業(yè)務(wù)錯(cuò)誤碼)提供給前端,前端再映射做”提示“。如用戶輸入非法請(qǐng)求,從邏輯上后端都沒(méi)法自己修復(fù),這是個(gè)“正確”的策略。

2 報(bào)500了嘞!

如用戶上傳一個(gè)頭像,后端將圖片發(fā)給[云存儲(chǔ)],結(jié)果云存儲(chǔ)報(bào)500,咋辦?你可能想重試,因?yàn)橐苍S僅是[網(wǎng)絡(luò)抖動(dòng)],重試就能正常執(zhí)行。但若重試多次無(wú)效,若設(shè)計(jì)了某種熱備方案,可能改為發(fā)到另一個(gè)服務(wù)器。“重試”和“使用備份的依賴”都是“立刻處理“。

但若重試無(wú)效,所有的[備份服務(wù)]也無(wú)效,也許就能像上面那樣把錯(cuò)誤拋給前端,提示用戶“服務(wù)器開(kāi)小差”。從這方案易看出,你想把錯(cuò)誤拋到哪里是因?yàn)槟莻€(gè)catch的地方是處理問(wèn)題最方便的地方。一個(gè)問(wèn)題的解決方案可能要幾個(gè)不同的錯(cuò)誤處理組合起來(lái)才能辦到。

3 NPE了!

你的程序拋個(gè)NPE。這一般就是程序員的bug:

要不就是程序員想表達(dá)一個(gè)東西”沒(méi)有“,結(jié)果在后續(xù)處理中忘判斷是否為null

要不就是在寫(xiě)代碼時(shí)覺(jué)得100%不可能為null的地方出現(xiàn)了一個(gè)null

不管哪種,這錯(cuò)誤用戶總會(huì)看到一個(gè)很含糊的報(bào)錯(cuò)信息,這遠(yuǎn)遠(yuǎn)不夠。“正確”辦法是程序員自己能盡快發(fā)現(xiàn)它,并盡快修復(fù)。要做到這點(diǎn),需要[監(jiān)控系統(tǒng)]不斷爬log,把問(wèn)題報(bào)警出來(lái)。而非等用戶找客服投訴。

4 OOM了!

比如你的[后端程序]突然OOM掛了。掛的程序沒(méi)法恢復(fù)自己。要做到“正確”,須在服務(wù)之外的容器考慮這問(wèn)題。

如你的服務(wù)跑在[k8s],他們會(huì)監(jiān)控你程序狀態(tài),然后重啟新的服務(wù)實(shí)例彌補(bǔ)掛掉的服務(wù),還得調(diào)整流量,把去往宕機(jī)服務(wù)的流量切換到新實(shí)例。這的恢復(fù)因?yàn)榭缦到y(tǒng)所以不能僅用異常實(shí)現(xiàn),但道理一樣。

但光靠重啟就“正確”了?若服務(wù)是完全無(wú)狀態(tài),問(wèn)題不大。但若有狀態(tài),部分用戶數(shù)據(jù)可能被執(zhí)行一半的請(qǐng)求搞亂。因此重啟要留意先“恢復(fù)數(shù)據(jù)到合法狀態(tài)”。這又回到你要知道咋樣才是“正確”的做法。只依靠簡(jiǎn)單的語(yǔ)法功能不能無(wú)腦解決這事。

5 提升維度

一個(gè)工作線程的“外部容器“是管理工作線程的“master”

一個(gè)網(wǎng)絡(luò)請(qǐng)求的“外部容器”是一個(gè)Web Server

一個(gè)用戶進(jìn)程的“外部容器”是[操作系統(tǒng)]

Erlang把這種supervisor-worker的機(jī)制融入到語(yǔ)言的設(shè)計(jì)

Web程序很大程度能把異常拋給頂層,是因?yàn)椋?/p>

請(qǐng)求來(lái)自前端,對(duì)因?yàn)橛脩粽?qǐng)求有誤(數(shù)據(jù)合法性、權(quán)限、用戶上下文狀態(tài))造成的問(wèn)題,最終基本只能告訴用戶。因此拋異常到一個(gè)集中處理錯(cuò)誤的地方,把異常轉(zhuǎn)換為某個(gè)業(yè)務(wù)錯(cuò)誤碼的方法,合理

后端服務(wù)一般無(wú)狀態(tài)。這也是軟件系統(tǒng)設(shè)計(jì)的一般原則。無(wú)狀態(tài)才意味著可隨時(shí)隨地安心重啟。用戶數(shù)據(jù)不會(huì)因?yàn)橐驗(yàn)橄乱粭l而會(huì)出問(wèn)題

后端對(duì)數(shù)據(jù)的修改依賴DB的事務(wù)。因此一個(gè)改一半的、沒(méi)提交的事務(wù)不會(huì)造成副作用。

但這3條件并非總成立??偰苡龅剑?/p>

一些處理邏輯并非無(wú)狀態(tài)

也并非所有的數(shù)據(jù)修改都能用一個(gè)事務(wù)保護(hù)

尤其要注意對(duì)[微服務(wù)]的調(diào)用,對(duì)內(nèi)存狀態(tài)「的修改是沒(méi)有事務(wù)保護(hù)的」 ,一不留神就會(huì)搞亂用戶數(shù)據(jù)。比如下面代碼段

6 難以排查的代碼段

try{
intres1=doStep1();
this.status1+=res1;
intres2=doStep2();
this.status2+=res2;
//拋個(gè)異常
intres3=doStep3();
this.status3=status1+status2+res3;
}catch(...){
//...
}

先假設(shè)status1、status2、status3之間需維護(hù)某種不變的約束(invariant)。然后執(zhí)行這段代碼時(shí),如在doStep3拋異常,下面對(duì)status3的賦值就不會(huì)執(zhí)行。這時(shí)如不能將status1、status2的修改rollback,就會(huì)造成數(shù)據(jù)違反約束的問(wèn)題。

而程序員很難發(fā)現(xiàn)這個(gè)數(shù)據(jù)被改壞了。壞數(shù)據(jù)還可能導(dǎo)致其他依賴這數(shù)據(jù)的代碼邏輯出錯(cuò)(如原本應(yīng)該給積分的,卻沒(méi)給)。而這種錯(cuò)誤一般很難排查,從大量數(shù)據(jù)里找到不正確的那一小段何其困難。

7 更難搞定的代碼段

//controller
voidcontrollerMethod(/*參數(shù)*/){
try{
returnsvc.doWorkAndGetResult(/*參數(shù)*/);
}catch(Exceptione){
returnErrorJsonObject.of(e);
}
}

//svc
voiddoWorkAndGetResult(/*someparams*/){
intres1=otherSvc1.doStep1(/*someparams*/);
this.status1+=res1;
intres2=otherSvc2.doStep2(/*someparams*/);
this.status2+=res2;
intres3=otherSvc3.doStep3(/*someparams*/);
this.status3=status1+status2+res3;
returnSomeResult.of(this.status1,this.status2,this.status3);
}

難搞在于你寫(xiě)的時(shí)候可能以為doStep1~3這種東西即使拋異常也能被Controller里的catch。

在svc這層是不用處理任何異常,「因此不寫(xiě)[try……catch]天經(jīng)地義」 。但實(shí)際上doStep1、doStep2、doStep3任何一個(gè)拋異常都會(huì)造成svc的數(shù)據(jù)狀態(tài)不一致。甚至你一開(kāi)始都可以通過(guò)文檔或其他溝通確定doStep1、doStep2、doStep3一開(kāi)始都是必然可成功,不會(huì)拋錯(cuò)的,因此你寫(xiě)的代碼一開(kāi)始是對(duì)的。

但你可能無(wú)法控制他們的實(shí)現(xiàn)(如他們是另外一個(gè)團(tuán)隊(duì)開(kāi)發(fā)的[jar]提供的),而他們的實(shí)現(xiàn)可能會(huì)改成拋錯(cuò)。你的代碼可能在完全不自知情況下從“不會(huì)出問(wèn)題”變成“可能出問(wèn)題”…… 更可怕的類似代碼不能正確工作:

voiddoWorkAndGetResult(/*someparams*/){
try{
intres1=otherSvc1.doStep1(/*someparams*/);
this.status1+=res1;
intres2=otherSvc2.doStep2(/*someparams*/);
this.status2+=res2;
intres3=otherSvc3.doStep3(/*someparams*/);
this.status3=status1+status2+res3;
returnSomeResult.of(this.status1,this.status2,this.status3);
}catch(Exceptione){
//dorollback
}
}

你以為這樣就會(huì)處理好數(shù)據(jù)rollback,甚至「覺(jué)得這種代碼優(yōu)雅」 。但實(shí)際上doStep1~3每一個(gè)地方拋錯(cuò),rollback的代碼都不一樣。

得這么寫(xiě)

voiddoWorkAndGetResult(/*someparams*/){
intres1,res2,res3;
try{
res1=otherSvc1.doStep1(/*someparams*/);
this.status1+=res1;
}catch(Exceptione){
throwe;
}

try{
res2=otherSvc2.doStep2(/*someparams*/);
this.status2+=res2;
}catch(Exceptione){
//rollbackstatus1
this.status1-=res1;
throwe;
}

try{
res3=otherSvc3.doStep3(/*someparams*/);
this.status3=status1+status2+res3;
}catch(Exceptione){
//rollbackstatus1&status2
this.status1-=res1;
this.status2-=res2;
throwe;
}
}

這才是得到正確結(jié)果的代碼,在任何地方出錯(cuò)都能維護(hù)數(shù)據(jù)一致性。優(yōu)雅嗎?

看起來(lái)很丑。比go的if err != nil還丑。但要在正確性和優(yōu)雅性取舍,肯定毫不猶豫選前者。作為程序員不能直接認(rèn)為拋異??山鉀Q任何問(wèn)題,須學(xué)會(huì)寫(xiě)出有正確邏輯的程序,哪怕很難且看起來(lái)丑。

為達(dá)成高正確性,你不能總將自己大部分注意力放在“一切都OK的流程“,而把錯(cuò)誤看作是可隨便應(yīng)付了事的工作或簡(jiǎn)單的相信exception可自動(dòng)搞定一切。

8 總結(jié)

希望程序員們對(duì)錯(cuò)誤處理都要有敬畏之心。Java因?yàn)镃hecked Exception設(shè)計(jì)問(wèn)題不得不避免使用,而Uncaughted Exception「實(shí)在是太過(guò)于弱雞,是不能給程序員提供更好地幫助的」 。

因此,程序員在每次拋錯(cuò)或者處理錯(cuò)誤的時(shí)候都要三省吾身:

這個(gè)錯(cuò)誤的處理是正確的嗎?

會(huì)讓用戶看到什么?

會(huì)不會(huì)搞亂數(shù)據(jù)?

不要以為自己拋了個(gè)異常就不管了。在[編譯器]不能幫上太多忙的時(shí)候,好好寫(xiě)UT來(lái)保護(hù)代碼脆弱的正確性。

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • JAVA
    +關(guān)注

    關(guān)注

    20

    文章

    2984

    瀏覽量

    106919
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4887

    瀏覽量

    70265
  • 程序員
    +關(guān)注

    關(guān)注

    4

    文章

    954

    瀏覽量

    30266
  • Service
    +關(guān)注

    關(guān)注

    0

    文章

    31

    瀏覽量

    14026

原文標(biāo)題:Service 層的異常是拋到 Controller 層還是直接處理?

文章出處:【微信號(hào):芋道源碼,微信公眾號(hào):芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    如何有效的處理空指針異常

    地遇到這個(gè)問(wèn)題。 那么我們應(yīng)該如何有效且優(yōu)雅的處理空指針異常呢? 下面了不起將詳細(xì)的介紹這個(gè)處理方案。 1、什么是空指針異常? 空指針異常
    的頭像 發(fā)表于 09-30 10:25 ?1869次閱讀

    嵌入式C編程常用的異常錯(cuò)誤處理

    。 3. 中斷服務(wù)程序 (Interrupt Service Routines, ISR) 在嵌入式系統(tǒng)中,中斷是處理異常情況的常用方法。ISR用于處理硬件中斷,并確保系統(tǒng)在
    發(fā)表于 08-06 14:32

    基于VxWorks的異常處理的研究和實(shí)現(xiàn)

    基于VxWorks的異常處理的研究和實(shí)現(xiàn)
    發(fā)表于 03-29 12:28 ?38次下載

    基于VxWorks的異常處理的研究和實(shí)現(xiàn)

    闡述了嵌入式軟件系統(tǒng)中異常處理的必要性,然后基于嵌入式實(shí)時(shí)操作系統(tǒng)VxWorks,介紹了一種與具體處理器類型無(wú)關(guān)的異常處理方法,并且結(jié)合一種
    發(fā)表于 01-11 09:13 ?23次下載

    java異常處理的設(shè)計(jì)與重構(gòu)

    在程序設(shè)計(jì)中,進(jìn)行異常處理是非常關(guān)鍵和重要的一部分。一個(gè)程序的異常處理框架的好壞直接影響到整個(gè)項(xiàng)目的代碼質(zhì)量以及后期維護(hù)成本和難度。試想一下,如果一個(gè)項(xiàng)目從頭到尾沒(méi)有考慮過(guò)
    發(fā)表于 09-27 15:40 ?1次下載
    java<b class='flag-5'>異常</b><b class='flag-5'>處理</b>的設(shè)計(jì)與重構(gòu)

    java異常處理設(shè)計(jì)和一些建議

    程序設(shè)計(jì)在程序設(shè)計(jì)中,進(jìn)行異常處理是非常關(guān)鍵和重要的一部分。一個(gè)程序的異常處理框架的好壞直接影響到整個(gè)項(xiàng)目的代碼質(zhì)量以及后期維護(hù)成本和難度。試想一下,如果一個(gè)項(xiàng)目從頭到尾沒(méi)有考慮過(guò)
    發(fā)表于 09-28 11:48 ?0次下載
    java<b class='flag-5'>異常</b><b class='flag-5'>處理</b>設(shè)計(jì)和一些建議

    C語(yǔ)言的異常處理案例代碼

    相信很多朋友在此之前可能根本沒(méi)有使用或者聽(tīng)說(shuō)過(guò)C語(yǔ)言的異常處理,印象中都是C++或者java才有的東西,C語(yǔ)言怎么會(huì)有異常處理呢?
    的頭像 發(fā)表于 12-22 08:44 ?3987次閱讀

    基于Python 異常的介紹以及異常處理的方法解析

    異常處理在任何一門(mén)編程語(yǔ)言里都是值得關(guān)注的一個(gè)話題,良好的異常處理可以讓你的程序更加健壯,清晰的錯(cuò)誤信息更能幫助你快速修復(fù)問(wèn)題。在Python中,和不分高級(jí)語(yǔ)言一樣,使用了try/ex
    的頭像 發(fā)表于 01-31 14:20 ?6524次閱讀
    基于Python <b class='flag-5'>異常</b>的介紹以及<b class='flag-5'>異常</b><b class='flag-5'>處理</b>的方法解析

    基于ARM處理器的高效異常處理解決方案

    嵌入式系統(tǒng)要求對(duì)異常及中斷處理器能快速響應(yīng)。文中分析了ARM體系結(jié)構(gòu)下 異常處理 特點(diǎn),提出一種基于 ARM處理器 的高效
    發(fā)表于 02-03 03:38 ?1571次閱讀
    基于ARM<b class='flag-5'>處理</b>器的高效<b class='flag-5'>異常</b><b class='flag-5'>處理</b>解決方案

    Java程序設(shè)計(jì)教程之異常處理的詳細(xì)資料說(shuō)明

    本文檔的詳細(xì)介紹的是Java程序設(shè)計(jì)教程之異常處理的詳細(xì)資料說(shuō)明主要內(nèi)容包括了:1 什么是異常,2異常處理機(jī)制,3
    發(fā)表于 02-22 10:27 ?13次下載
    Java程序設(shè)計(jì)教程之<b class='flag-5'>異常</b><b class='flag-5'>處理</b>的詳細(xì)資料說(shuō)明

    ARM異常中斷的原因及處理措施

    當(dāng)ARM異常中斷發(fā)生時(shí),系統(tǒng)執(zhí)行完當(dāng)前指令后,將跳轉(zhuǎn)到相應(yīng)的異常中斷處理程序處執(zhí)行。當(dāng)異常中斷處理程序執(zhí)行完成后,程序返回到發(fā)生中斷指令的下
    的頭像 發(fā)表于 06-17 10:05 ?8527次閱讀

    處理器中異常和中斷解決

    異常是能夠引起程序流偏離正常流程的事件,當(dāng)異常發(fā)生時(shí),正在執(zhí)行的程序就會(huì)被掛起,處理器轉(zhuǎn)而執(zhí)行一塊與該事件相關(guān)的代碼(異常處理)。事件可以是
    的頭像 發(fā)表于 10-12 17:14 ?5455次閱讀

    替代try catch處理異常的優(yōu)雅方式

    不過(guò)跟異常處理相關(guān)的只有注解@ExceptionHandler,從字面上看,就是 異常處理器 的意思,其實(shí)際作用也是:若在某個(gè)Controller類定義一個(gè)
    的頭像 發(fā)表于 10-26 10:18 ?1278次閱讀

    C++程序異常處理機(jī)制是什么

    那么C++設(shè)計(jì)了一套異常處理機(jī)制,一方面能夠使得異常處理和正常運(yùn)行代碼進(jìn)行分離,使得程序更加模塊化;另一方面,C++的異常
    的頭像 發(fā)表于 02-21 10:37 ?1089次閱讀
    C++程序<b class='flag-5'>異常</b><b class='flag-5'>處理</b>機(jī)制是什么

    PLC的異常類型和處理辦法

    1.中央處理異常: 如果出現(xiàn)中央處理異常報(bào)警,應(yīng)檢查連接到中央處理器內(nèi)部總線的所有設(shè)備。具體方法是依次更換可能導(dǎo)致故障的機(jī) 組,找出故障
    發(fā)表于 04-19 09:43 ?0次下載
    PLC的<b class='flag-5'>異常</b>類型和<b class='flag-5'>處理</b>辦法

    電子發(fā)燒友

    中國(guó)電子工程師最喜歡的網(wǎng)站

    • 2931785位工程師會(huì)員交流學(xué)習(xí)
    • 獲取您個(gè)性化的科技前沿技術(shù)信息
    • 參加活動(dòng)獲取豐厚的禮品