改狀態,你真的會改嗎?

来源:https://www.cnblogs.com/buguge/archive/2020/03/23/12554965.html
-Advertisement-
Play Games

企業應用中,涉及到修改狀態的場景太多了。比如,企業入網後,要審核資質。個人領取任務後,企業管理員要審核領取人。 應用管理系統中,通常是下圖這樣,在列表後有操作按鈕來修改數據記錄的狀態。 點擊“通過”/“拒絕”操作,要修改數據記錄的status欄位。服務端程式邏輯怎麼實現呢? 先定義服務端api介面: ...


企業應用中,涉及到修改狀態的場景太多了。比如,企業入網後,要審核資質。個人領取任務後,企業管理員要審核領取人。

應用管理系統中,通常是下圖這樣,在列表後有操作按鈕來修改數據記錄的狀態。

點擊“通過”/“拒絕”操作,要修改數據記錄的status欄位。服務端程式邏輯怎麼實現呢?

 

先定義服務端api介面:

  1. Request URL: http://***/api/enterprise/taskApply/audit
  2. Request Method: POST
  3. Content-Type: application/json;charset=UTF-8
  4. User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/80.0.3987.149 Safari/537.36
  5. Request Payload: {"applyIds":"1,2,3","auditStatus":"PASS"}

 

系統採用前後端分離,系統架構是分散式的。SpringMVC的RestController通過遠程調用Dubbo介面,來實現數據的CRUD。

項目使用jeecg-boot逆向工程生成代碼,然後在此基礎上進行調整。

 

程式邏輯v1

    @RequestMapping(value = "/enterprise/taskApply/audit",method = RequestMethod.POST)
    public Result audit(@RequestBody TaskApplyVO taskApplyVO, HttpServletRequest req){
        //當前登錄企業
        EnterpriseVO enterpriseVo = EnterpriseContext.getEnterpriseVo();
        if(enterpriseVo == null){
            return Result.error("未登錄");
        }
        if(StringUtils.isEmpty(taskApplyVO.getApplyIds())){
            return Result.error("未選擇");
        }
        String[] applyIds = taskApplyVO.getApplyIds().split(",");
        for (String applyId : applyIds) {
            TaskApplyVO taskApply = new TaskApplyVO();
            taskApply.setApplyId(Long.parseLong(applyId));
            taskApply.setStatus(taskApplyVO.getStatus());
            taskApply.setAuditTime(DateUtils.getDate());
            taskApplyService.updateById(taskApply);
        }

        return Result.ok("成功");
    }

 

其中,taskApplyService是遠程RPC介面實例。

系統使用了一段時間後,bug出現了————已經審核完了的記錄還能再審核。什麼情況下會出現這種情況呢?我們且不說。不過看代碼邏輯,我們可以發現,在修改一條記錄的狀態欄位之前並沒有判斷狀態欄位的初始值(在 待審核 狀態下 才能修改為 審核通過or審核拒絕),所以會出現這種情況。

程式邏輯v2

修複上面的bug。加上前置狀態判斷。

幸好是這種常見的業務處理,fix掉即可。如果是在支付系統里,付款成功的交易還能被改成付款失敗,那可就出現資金損失了,哭都沒地兒哭去。

    @RequestMapping(value = "/enterprise/taskApply/audit",method = RequestMethod.POST)
    public Result audit(@RequestBody TaskApplyVO taskApplyVO, HttpServletRequest req){
        //當前登錄企業
        EnterpriseVO enterpriseVo = EnterpriseContext.getEnterpriseVo();
        if(enterpriseVo == null){
            return Result.error("未登錄");
        }
        if(StringUtils.isEmpty(taskApplyVO.getApplyIds())){
            return Result.error("未選擇");
        }
        String[] applyIds = taskApplyVO.getApplyIds().split(",");
        for (String applyId : applyIds) {
            TaskApplyVO taskApply = taskApplyService.getById(applyId);
            if(!TaskApplyStatusEnum.TO_AUDIT.name().equals(taskApply.getStatus())){
                continue;
            }

            taskApply.setStatus(taskApplyVO.getStatus());
            taskApply.setAuditTime(DateUtils.getDate());
            taskApplyService.updateById(taskApply);
        }

        return Result.ok("成功");
    }

 

系統使用了一段時間,bug出現了————用戶對數據記錄的修改莫名其妙的丟失了。哈哈,看上面的代碼,併發較多下,對這張數據表的同一條記錄的多個不同操作請求都出現時,比如這裡是審核,同時還有信息修改,是不是會出現覆蓋的情況?是的,因為在rpc調用getById與rpc調用updateById之間是有時間間隔的。兩個線程都通過getById取到了數據記錄,然後都修改了不同的欄位後執行updateById,資料庫操作本身是有先後的,所以,可能就會出現其中一次update覆蓋另一次update。那怎麼改呢?用分散式事務來控制?那可就比較費事了。為了減少這種衝突發生的可能性,還是先重構一下我們這個方法的邏輯吧。

程式邏輯v3

修改上面邏輯,new一個實體對象,只update所需欄位。

    @RequestMapping(value = "/enterprise/taskApply/audit",method = RequestMethod.POST)
    public Result audit(@RequestBody TaskApplyVO taskApplyVO, HttpServletRequest req){
        //當前登錄企業
        EnterpriseVO enterpriseVo = EnterpriseContext.getEnterpriseVo();
        if(enterpriseVo == null){
            return Result.error("未登錄");
        }
        if(StringUtils.isEmpty(taskApplyVO.getApplyIds())){
            return Result.error("未選擇");
        }
        String[] applyIds = taskApplyVO.getApplyIds().split(",");
        for (String applyId : applyIds) {
            TaskApplyVO byId = taskApplyService.getById(applyId);
            if(!TaskApplyStatusEnum.TO_AUDIT.name().equals(byId.getStatus())){
                continue;
            }
            TaskApplyVO taskApply = new TaskApplyVO();
            taskApply.setApplyId(Long.parseLong(applyId));
            taskApply.setStatus(taskApplyVO.getStatus());
            taskApply.setAuditTime(DateUtils.getDate());
            taskApplyService.updateById(taskApply);
        }

        return Result.ok("保存成功");
    }

 

系統使用了一段時間,bug出現了。什麼bug?還是最開始的bug————已經審核完了的記錄還能再審核。 !!!奔潰了~~

程式邏輯v4

使用狀態鎖。

同時,變更程式實現。將審核的邏輯後置封裝到RPC服務里。好的程式設計往往是把邏輯封裝起來,利於維護,也利於復用。

    @RequestMapping(value = "/enterprise/taskApply/audit",method = RequestMethod.POST)
    public Result audit(@RequestBody TaskApplyVO taskApplyVO, HttpServletRequest req){
        //當前登錄企業
        EnterpriseVO enterpriseVo = EnterpriseContext.getEnterpriseVo();
        if(enterpriseVo == null){
            return Result.error("未登錄");
        }
        if(StringUtils.isEmpty(taskApplyVO.getApplyIds())){
            return Result.error("未選擇");
        }
        String[] applyIds = taskApplyVO.getApplyIds().split(",");
        return taskApplyService.audit(applyIds, taskApplyVO.getStatus());
    }

下麵是TaskApplyService實現類中的audit方法。可以看出來,每次迴圈還少了一次讀庫查詢。

    @Override
    public Result audit(String[] taskIdArr, String auditStatus) {
        for (String id : taskIdArr) {
            TaskApply taskApply = new TaskApply();
            taskApply.setStatus(auditStatus);
            taskApply.setAuditTime(DateUtils.getDate());
            taskApplyManager.update(taskApply, new LambdaQueryWrapper<TaskApply>()
                    .eq(TaskApply::getApplyId, Long.parseLong(id))
                    .eq(TaskApply::getStatus, TaskApplyStatusEnum.TO_AUDIT.name()));
        }
        return Result.ok();
    }

 


我們修改mybatis-plus配置,把程式執行的sql列印出來
mybatis-plus:
  configuration:
    # 這個配置會將執行的sql列印出來,在開發或測試的時候可以用
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
如下是程式執行的sql。where條件里有status欄位。update執行成功返回1,否則返回0。
==>  Preparing: UPDATE emax_task_apply SET apply_status=?, audit_time=? WHERE apply_id = ? AND status = ? 
==> Parameters: TASKAPPLY_PASS(String), 2020-03-23 16:52:44.186(Timestamp), 1(Long), TO_AUDIT(String)
<==    Updates: 1

 

 

 結束

感謝閱讀,本文整理用時2:51:00(18:00~20:51)。不足之處,歡迎交流!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 轉載聲明 本系列文章轉自某技術大佬的博客https://www.cnblogs.com/bangerlee/ 該系列文章是我在網上能夠找到的最全面的分散式理論介紹文章了,一直沒看到有人整理這個系列文章,所以這次我就來做技術好文的搬運工,給整合了一把,覺得寫得好的朋友不妨去這位大佬的博客上打賞一把。 ...
  • Go中存在著不少內置函數,此類函數並不需要引入相關Package就可以直接使用該類函數。在Go的源碼builtin包的builtin.go中定義Go所有的內置函數;但該文件僅僅是定義描述出了所有內置函數,並不包含函數的任何實現代碼,該文件除了定義了內置函數還定義了部分內置類型; 內置函數使用 len ...
  • Java 7 為懶惰的Java開發人員帶來了一些非常好的功能。 嘗試資源 是這種功能之一,它可以減少代碼行,並使代碼更健壯。在本教程中,我將討論有關此功能的內容。 1. 資源清除的舊方法(在Java 7之前) 我們長期以來一直在這樣做。例如,從文件系統讀取文件。代碼可能看起來有所不同,但流程如下例所 ...
  • 承接上文 通過discovery和define完成了第一輪企業級別的發散和收斂。 即:站在企業的高度,基於企業願景和內外部環境,通過戰略分解和現狀調研,應用企業架構的方法確定了最終的平臺型企業架構,並確定了需要哪些中台,以及建設先後的問題。 中台的設計階段:進行第二輪的發散和收斂,站在一個中台產品的 ...
  • 在Java 7發行版中,oracle在異常處理機制上也做了一些不錯的更改。這些主要是 改進的catch塊 和 多餘的throws子句 。讓我們看看他們是如何改變的。 1.改進了Java 7中的catch塊 在此功能中,現在您可以 在單個catch塊中捕獲多個異常 。在Java 7之前,您只能在每個c ...
  • 入門訓練3 圓的面積 問題描述 給定圓的半徑r,求圓的面積。 輸入格式 輸入包含一個整數r,表示圓的半徑。 輸出格式 輸出一行,包含一個實數,四捨五入保留小數點後7位,表示圓的面積。 說明:在本題中,輸入是一個整數,但是輸出是一個實數。 對於實數輸出的問題,請一定看清楚實數輸出的要求,比如本題中要求 ...
  • 第一章 認識Java8以及函數式編程 儘管距離Java8發佈已經過去7、8年的時間,但時至今日仍然有許多公司、項目停留在Java7甚至更早的版本。即使已經開始使用Java8的項目,大多數程式員也仍然採用“傳統”的編碼方式。 即使是在Java7就已經有了處理異常的新方式—— ,但大多數程式員也仍然採用 ...
  • 前言 2019 年底開始蔓延的新型肺炎疫情牽動人心,作為個體,我們力所能及的就是儘量待在家中少出門。 看到一些朋友叫設計同學幫忙給自己的頭像戴上口罩,作為技術人,心想一定還有更多人有這樣的訴求,不如開發一個簡單的程式來實現這個需求,也算是幫助設計姐姐減少工作量。 於是花了些時間,寫了一個叫做 fac ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...