改狀態,你真的會改嗎?

来源: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
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...