java服務端實踐:防止訂單重複提交支付

來源:cnblogs.com/cjsblog/p/14516909.html


  • 概述
  • 爲了防止掉單,這裏可以這樣處理:
  • 爲了防止訂單重複提交,可以這樣處理:
  • 附上微信支付最佳實踐:

概述

如圖是一個簡化的下單流程,首先是提交訂單,然後是支付。

支付的話,一般是走支付網關(支付中心),然後支付中心與第三方支付渠道(微信、支付寶、銀聯)交互。

支付成功以後,異步通知支付中心,支付中心更新自身支付訂單狀態,再通知業務應用,各業務再更新各自訂單狀態。

這個過程中經常可能遇到的問題是掉單,無論是超時未收到回調通知也好,還是程序自身報錯也好。

總之由於各種各樣的原因,沒有如期收到通知並正確的處理後續邏輯等等,都會造成用戶支付成功了,但是服務端這邊訂單狀態沒更新。

這個時候有可能產生投訴,或者用戶重複支付。

由於③⑤造成的掉單稱之爲外部掉單,由④⑥造成的掉單我們稱之爲內部掉單

爲了防止掉單,這裏可以這樣處理:

1、支付訂單增加一箇中間狀態“支付中”,當同一個訂單去支付的時候,先檢查有沒有狀態爲“支付中”的支付流水,當然支付(prepay)的時候要加個鎖。支付完成以後更新支付流水狀態的時候再講其改成“支付成功”狀態。

2、支付中心這邊要自己定義一個超時時間(比如:30秒),在此時間範圍內如果沒有收到支付成功回調,則應調用接口主動查詢支付結果,比如10s、20s、30s查一次,如果在最大查詢次數內沒有查到結果,應做異常處理

3、支付中心收到支付結果以後,將結果同步給業務系統,可以發MQ,也可以直接調用,直接調用的話要加重試(比如:SpringBoot Retry)

4、無論是支付中心,還是業務應用,在接收支付結果通知時都要考慮接口冪等性,消息只處理一次,其餘的忽略

5、業務應用也應做超時主動查詢支付結果

對於上面說的超時主動查詢可以在發起支付的時候將這些支付訂單放到一張表中,用定時任務去掃

爲了防止訂單重複提交,可以這樣處理:

1、創建訂單的時候,用訂單信息計算一個哈希值,判斷redis中是否有key,有則不允許重複提交,沒有則生成一個新key,放到redis中設置個過期時間,然後創建訂單。

其實就是在一段時間內不可重複相同的操作

附上微信支付最佳實踐:



往期推薦



應用卡死的bug竟然是因爲Redis使用不當?

SpringBoot+webSocket實現掃碼登錄功能

7000+字,30+張圖!Java線上故障排查思路全部總結在這裏了

TCP3次握手爲啥揮手卻要4次,這下解釋明白了

SpringCloud微服務的熔斷機制和熔斷的意義?

java項目線上JVM調優實踐,FullGC大大減少

非常nice的分佈式事務方案總結

【乾貨】6500字全面字講解 Redis 性能優化點!

精髓!深入淺出剖析12個Zookeeper知識點

Spring+SpringMVC+Mybatis分佈式敏捷開發系統架構(附源碼)

Java大文件HTTP斷點續傳到服務器該怎麼做?

面試官:請你講講Thread.sleep(0) 的作用?


本文分享自微信公衆號 - 俠夢的開發筆記(xmdevnote)。
如有侵權,請聯繫 [email protected] 刪除。
本文參與“OSC源創計劃”,歡迎正在閱讀的你也加入,一起分享。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章