精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Service層的異常處理

jf_ro2CN3Fa ? 來源:JavaEdge ? 2024-01-08 11:29 ? 次閱讀

來源:JavaEdge

0 前言

一般初學者學習編碼和[錯誤處理]時,先知道[編程語言]有一種處理錯誤的形式或約定(如Java就拋異常),然后就開始用這些工具。但卻忽視這問題本質:「處理錯誤是為了寫正確程序」 。可是

1 啥叫“正確”?

「由解決的問題決定的。問題不同,解決方案不同。」

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

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

2 報500了嘞!

如用戶上傳一個頭像,后端將圖片發給[云存儲],結果云存儲報500,咋辦?你可能想重試,因為也許僅是[網絡抖動],重試就能正常執行。但若重試多次無效,若設計了某種熱備方案,可能改為發到另一個服務器。“重試”和“使用備份的依賴”都是“立刻處理“。

但若重試無效,所有的[備份服務]也無效,也許就能像上面那樣把錯誤拋給前端,提示用戶“服務器開小差”。從這方案易看出,你想把錯誤拋到哪里是因為那個catch的地方是處理問題最方便的地方。一個問題的解決方案可能要幾個不同的錯誤處理組合起來才能辦到。

3 NPE了!

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

要不就是程序員想表達一個東西”沒有“,結果在后續處理中忘判斷是否為null

要不就是在寫代碼時覺得100%不可能為null的地方出現了一個null

不管哪種,這錯誤用戶總會看到一個很含糊的報錯信息,這遠遠不夠。“正確”辦法是程序員自己能盡快發現它,并盡快修復。要做到這點,需要[監控系統]不斷爬log,把問題報警出來。而非等用戶找客服投訴。

4 OOM了!

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

如你的服務跑在[k8s],他們會監控你程序狀態,然后重啟新的服務實例彌補掛掉的服務,還得調整流量,把去往宕機服務的流量切換到新實例。這的恢復因為跨系統所以不能僅用異常實現,但道理一樣。

但光靠重啟就“正確”了?若服務是完全無狀態,問題不大。但若有狀態,部分用戶數據可能被執行一半的請求搞亂。因此重啟要留意先“恢復數據到合法狀態”。這又回到你要知道咋樣才是“正確”的做法。只依靠簡單的語法功能不能無腦解決這事。

5 提升維度

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

一個網絡請求的“外部容器”是一個Web Server

一個用戶進程的“外部容器”是[操作系統]

Erlang把這種supervisor-worker的機制融入到語言的設計

Web程序很大程度能把異常拋給頂層,是因為:

請求來自前端,對因為用戶請求有誤(數據合法性、權限、用戶上下文狀態)造成的問題,最終基本只能告訴用戶。因此拋異常到一個集中處理錯誤的地方,把異常轉換為某個業務錯誤碼的方法,合理

后端服務一般無狀態。這也是軟件系統設計的一般原則。無狀態才意味著可隨時隨地安心重啟。用戶數據不會因為因為下一條而會出問題

后端對數據的修改依賴DB的事務。因此一個改一半的、沒提交的事務不會造成副作用。

但這3條件并非總成立。總能遇到:

一些處理邏輯并非無狀態

也并非所有的數據修改都能用一個事務保護

尤其要注意對[微服務]的調用,對內存狀態「的修改是沒有事務保護的」 ,一不留神就會搞亂用戶數據。比如下面代碼段

6 難以排查的代碼段

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

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

而程序員很難發現這個數據被改壞了。壞數據還可能導致其他依賴這數據的代碼邏輯出錯(如原本應該給積分的,卻沒給)。而這種錯誤一般很難排查,從大量數據里找到不正確的那一小段何其困難。

7 更難搞定的代碼段

//controller
voidcontrollerMethod(/*參數*/){
try{
returnsvc.doWorkAndGetResult(/*參數*/);
}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);
}

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

在svc這層是不用處理任何異常,「因此不寫[try……catch]天經地義」 。但實際上doStep1、doStep2、doStep3任何一個拋異常都會造成svc的數據狀態不一致。甚至你一開始都可以通過文檔或其他溝通確定doStep1、doStep2、doStep3一開始都是必然可成功,不會拋錯的,因此你寫的代碼一開始是對的。

但你可能無法控制他們的實現(如他們是另外一個團隊開發的[jar]提供的),而他們的實現可能會改成拋錯。你的代碼可能在完全不自知情況下從“不會出問題”變成“可能出問題”…… 更可怕的類似代碼不能正確工作:

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

你以為這樣就會處理好數據rollback,甚至「覺得這種代碼優雅」 。但實際上doStep1~3每一個地方拋錯,rollback的代碼都不一樣。

得這么寫

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

這才是得到正確結果的代碼,在任何地方出錯都能維護數據一致性。優雅嗎?

看起來很丑。比go的if err != nil還丑。但要在正確性和優雅性取舍,肯定毫不猶豫選前者。作為程序員不能直接認為拋異常可解決任何問題,須學會寫出有正確邏輯的程序,哪怕很難且看起來丑。

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

8 總結

希望程序員們對錯誤處理都要有敬畏之心。Java因為Checked Exception設計問題不得不避免使用,而Uncaughted Exception「實在是太過于弱雞,是不能給程序員提供更好地幫助的」

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

這個錯誤的處理是正確的嗎?

會讓用戶看到什么?

會不會搞亂數據?

不要以為自己拋了個異常就不管了。在[編譯器]不能幫上太多忙的時候,好好寫UT來保護代碼脆弱的正確性。

審核編輯:湯梓紅
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • JAVA
    +關注

    關注

    19

    文章

    2960

    瀏覽量

    104562
  • 代碼
    +關注

    關注

    30

    文章

    4753

    瀏覽量

    68368
  • 程序員
    +關注

    關注

    4

    文章

    950

    瀏覽量

    29768
  • Service
    +關注

    關注

    0

    文章

    30

    瀏覽量

    13774

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

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

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

    地遇到這個問題。 那么我們應該如何有效且優雅的處理空指針異常呢? 下面了不起將詳細的介紹這個處理方案。 1、什么是空指針異常? 空指針異常
    的頭像 發表于 09-30 10:25 ?1465次閱讀

    嵌入式C編程常用的異常錯誤處理

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

    基于VxWorks的異常處理的研究和實現

    基于VxWorks的異常處理的研究和實現
    發表于 03-29 12:28 ?38次下載

    基于VxWorks的異常處理的研究和實現

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

    java異常處理的設計與重構

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

    java異常處理設計和一些建議

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

    C語言的異常處理案例代碼

    相信很多朋友在此之前可能根本沒有使用或者聽說過C語言的異常處理,印象中都是C++或者java才有的東西,C語言怎么會有異常處理呢?
    的頭像 發表于 12-22 08:44 ?3799次閱讀

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

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

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

    嵌入式系統要求對異常及中斷處理器能快速響應。文中分析了ARM體系結構下 異常處理 特點,提出一種基于 ARM處理器 的高效
    發表于 02-03 03:38 ?1385次閱讀
    基于ARM<b class='flag-5'>處理</b>器的高效<b class='flag-5'>異常</b><b class='flag-5'>處理</b>解決方案

    Java程序設計教程之異常處理的詳細資料說明

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

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

    當ARM異常中斷發生時,系統執行完當前指令后,將跳轉到相應的異常中斷處理程序處執行。當異常中斷處理程序執行完成后,程序返回到發生中斷指令的下
    的頭像 發表于 06-17 10:05 ?8027次閱讀

    處理器中異常和中斷解決

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

    替代try catch處理異常的優雅方式

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

    C++程序異常處理機制是什么

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

    PLC的異常類型和處理辦法

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