自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

一個(gè)注解實(shí)現(xiàn)接口冪等,這樣才優(yōu)雅!

開(kāi)發(fā) 前端
本節(jié)內(nèi)容介紹了防重注解@RepeatSubmit的實(shí)現(xiàn)原理,后續(xù)開(kāi)發(fā)中只需要在非查詢(xún)接口中添加這個(gè)注解就能保證在一定時(shí)間內(nèi)防止重復(fù)提交。

場(chǎng)景

碼猿慢病云管理系統(tǒng)中其實(shí)高并發(fā)的場(chǎng)景不是很多,沒(méi)有必要每個(gè)接口都去考慮并發(fā)高的場(chǎng)景,比如添加住院患者的這個(gè)接口,具體的業(yè)務(wù)代碼就不貼了,業(yè)務(wù)偽代碼如下:

圖片圖片

上述代碼有問(wèn)題嗎?誰(shuí)能說(shuō)有問(wèn)題?一般情況下是沒(méi)什么問(wèn)題,但是在高并發(fā)的場(chǎng)景下肯定是存在問(wèn)題,為什么?

因?yàn)橛惺聞?wù)的隔離性,step1這個(gè)階段對(duì)住院號(hào)的校驗(yàn)肯定是存在問(wèn)題的,在高并發(fā)的場(chǎng)景下無(wú)法保證這里的校驗(yàn)一定準(zhǔn)確。

其實(shí)這個(gè)接口的并發(fā)并不高,在碼猿慢病云管理系統(tǒng)中一般不會(huì)出現(xiàn)這種問(wèn)題,那么什么時(shí)候會(huì)出現(xiàn)呢?

醫(yī)院中大部分是內(nèi)網(wǎng)+外網(wǎng),如果由于網(wǎng)絡(luò)的抖動(dòng),系統(tǒng)請(qǐng)求響應(yīng)的時(shí)間延遲,這樣會(huì)導(dǎo)致醫(yī)護(hù)操作時(shí)會(huì)出現(xiàn)重復(fù)點(diǎn)擊的情況,比如1秒中之內(nèi)由于第一次點(diǎn)添加患者這個(gè)按鈕沒(méi)反應(yīng),往往護(hù)士都會(huì)重復(fù)點(diǎn)擊,這種情況下是會(huì)出現(xiàn)問(wèn)題。

這里我們就暫且不談對(duì)單個(gè)接口的冪等優(yōu)化了,要想一個(gè)方案全局解決這個(gè)問(wèn)題,在碼猿慢病云管理系統(tǒng)中其實(shí)只要保證這種并發(fā)不高的接口在一定時(shí)間段內(nèi)保證冪等即可,比如5秒之內(nèi),這樣在5秒之內(nèi)護(hù)士重復(fù)點(diǎn)擊就沒(méi)事。

解決方案

在碼猿慢病云管理系統(tǒng)中新增了一個(gè)注解:@RepeatSubmit,代碼如下:

圖片圖片

只需要將該注解標(biāo)注在新增、修改、刪除接口上就能保證在默認(rèn)的5秒之內(nèi)接口冪等。

比如新增住院患者這個(gè)接口:

圖片圖片

那么原理是什么?其實(shí)很簡(jiǎn)單,先來(lái)說(shuō)下原理,再介紹具體的實(shí)現(xiàn):

  1. AOP攔截增強(qiáng)@RepeatSubmit注解
  2. 獲取請(qǐng)求的URL、IP地址、請(qǐng)求參數(shù)
  3. 將請(qǐng)求URL、IP地址、請(qǐng)求參數(shù)以一定形式轉(zhuǎn)為key
  4. 借助Redis的setNx命令將key存入Redis,且設(shè)置失效時(shí)間
  5. 如果存入成功則允許訪問(wèn),失敗則拋出異常
  6. 全局異常捕獲,輸出指定信息給客戶(hù)端

上述6個(gè)步驟中其實(shí)只有一點(diǎn)比較難實(shí)現(xiàn)的,其他的都是基本操作,就是獲取這個(gè)請(qǐng)求參數(shù),下面將詳細(xì)介紹一下如何獲取這個(gè)請(qǐng)求參數(shù)。

獲取請(qǐng)求參數(shù)

對(duì)于form-data的入?yún)⒅恍枰{(diào)用HttpServletRequest的API讀取,但是對(duì)于@RequestBody標(biāo)注的入?yún)⑹峭ㄟ^(guò)IO流讀取數(shù)據(jù),且IO流只能被讀取一次,如果在AOP中讀取了,那么在接口層面的入?yún)⒆x取肯定是有問(wèn)題,報(bào)錯(cuò)如下:

圖片圖片

解決方案也很簡(jiǎn)單,只需要保證IO流能夠多次讀取即可,下面就來(lái)介紹一下方案。

這里我們可以利用裝飾者模式對(duì) HttpServletRequest 的功能進(jìn)行增強(qiáng),具體做法也很簡(jiǎn)單,我們重新定義一個(gè) HttpServletRequest:

圖片圖片

圖片圖片

這段代碼并不難,很好懂。

首先在構(gòu)造 RepeatedlyRequestWrapper 的時(shí)候,就通過(guò) IO 流將數(shù)據(jù)讀取出來(lái)并存入到一個(gè) byte 數(shù)組中,然后重寫(xiě) getReader 和 getInputStream 方法,在這兩個(gè)讀取 IO 流的方法中,都從 byte 數(shù)組中返回 IO 流數(shù)據(jù)出來(lái),這樣就實(shí)現(xiàn)了反復(fù)讀取了。

接下來(lái)我們定義一個(gè)過(guò)濾器,讓這個(gè)裝飾后的 Request 生效:

圖片圖片

判斷一下,如果請(qǐng)求數(shù)據(jù)類(lèi)型是 JSON 的話,就把 HttpServletRequest “偷梁換柱”改為 HttpRequestWrapper,然后讓過(guò)濾器繼續(xù)往下走。

這樣就可以配置后就可以在程序中反復(fù)讀取參數(shù)了!

防重注解實(shí)現(xiàn)

解決了參數(shù)讀取的問(wèn)題,下面就可以輕松實(shí)現(xiàn)這個(gè)防重注解了,首先定義注解com.code.ape.codeape.common.security.annotation.RepeatSubmit:

圖片圖片

接下來(lái)直接用AOP實(shí)現(xiàn),com.code.ape.codeape.common.security.component.CodeapeRepeatSubmitAspect代碼如下:

圖片圖片

圖片圖片

邏輯很簡(jiǎn)單,上述已經(jīng)介紹過(guò)完整的流程,這里需要注意的是參數(shù)的讀取,代碼如下:

圖片圖片

其實(shí)就是將request判斷下是否是經(jīng)過(guò)過(guò)濾器封裝后的HttpRequestWrapper對(duì)象,如果是的話則是@RequestBody入?yún)ⅲ苯訌腎O流中讀取。

總結(jié)

本節(jié)內(nèi)容介紹了防重注解@RepeatSubmit的實(shí)現(xiàn)原理,后續(xù)開(kāi)發(fā)中只需要在非查詢(xún)接口中添加這個(gè)注解就能保證在一定時(shí)間內(nèi)防止重復(fù)提交。

碼猿慢病云管理系統(tǒng)已經(jīng)在星球中陸續(xù)更新,目前更新內(nèi)容如下:

前言
     01 項(xiàng)目架構(gòu)+業(yè)務(wù)介紹
     02 三方組件介紹
     03 服務(wù)端項(xiàng)目部署
     04 前端項(xiàng)目部署
     05 多租戶(hù)架構(gòu)設(shè)計(jì)
     06 醫(yī)療系統(tǒng)中的權(quán)限如何設(shè)計(jì)?
     07 項(xiàng)目搭建
     08 關(guān)掉驗(yàn)證碼登錄
     09 開(kāi)發(fā)平臺(tái)自動(dòng)生成業(yè)務(wù)代碼
認(rèn)證鑒權(quán)
     01 認(rèn)證登錄生成token
     02 token檢驗(yàn)、鑒權(quán)
     03 token有效期設(shè)置
     04 刷新token
     05 檢查token
     06 服務(wù)中如何獲取當(dāng)前登錄用戶(hù)信息?
     07 接口對(duì)外暴露
     08 接口只允許內(nèi)部調(diào)用怎么處理?
     09 如何實(shí)現(xiàn)token中繼?
     10 當(dāng)前登錄用戶(hù)身份信息如何異步傳遞?
     11 科室權(quán)限如何定一個(gè)注解自動(dòng)注入?
     12 一個(gè)注解防止接口重復(fù)提交
業(yè)務(wù)
     01 科室管理
     02 醫(yī)院管理
     03 角色管理

責(zé)任編輯:武曉燕 來(lái)源: 碼猿技術(shù)專(zhuān)欄
相關(guān)推薦

2025-02-23 08:00:00

冪等性Java開(kāi)發(fā)

2024-04-07 08:48:13

WebSocket集群MQ

2024-11-08 15:56:36

2022-06-21 14:44:38

接口數(shù)據(jù)脫敏

2024-11-07 10:55:26

2022-06-10 13:03:44

接口重試while

2024-03-13 15:18:00

接口冪等性高并發(fā)

2021-04-20 10:50:38

Spring Boot代碼Java

2022-04-25 11:26:16

開(kāi)發(fā)SpringBoot

2022-05-16 10:45:22

Redis接口限流緩存

2023-08-21 08:01:03

2021-01-18 14:34:59

冪等性接口客戶(hù)端

2022-12-12 08:14:47

2021-01-04 09:12:31

集合變量

2024-08-29 09:01:39

2024-06-24 01:00:00

2023-03-07 08:19:16

接口冪等性SpringBoot

2025-03-11 08:20:58

2020-11-03 16:00:33

API接口微服務(wù)框架編程語(yǔ)言

2022-12-13 09:19:06

高并發(fā)SpringBoot
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)