在微服務(wù)架構(gòu)下,我們在完成一個訂單流程時經(jīng)常遇到下面的場景:

          * 一個訂單創(chuàng)建接口,第一次調(diào)用超時了,然后調(diào)用方重試了一次
          * 在訂單創(chuàng)建時,我們需要去扣減庫存,這時接口發(fā)生了超時,調(diào)用方重試了一次
          * 當這筆訂單開始支付,在支付請求發(fā)出之后,在服務(wù)端發(fā)生了扣錢操作,接口響應(yīng)超時了,調(diào)用方重試了一次
          * 一個訂單狀態(tài)更新接口,調(diào)用方連續(xù)發(fā)送了兩個消息,一個是已創(chuàng)建,一個是已付款。但是你先接收到已付款,然后又接收到了已創(chuàng)建
          * 在支付完成訂單之后,需要發(fā)送一條短信,當一臺機器接收到短信發(fā)送的消息之后,處理較慢。消息中間件又把消息投遞給另外一臺機器處理
          以上問題,就是在單體架構(gòu)轉(zhuǎn)成微服務(wù)架構(gòu)之后,帶來的問題。當然不是說單體架構(gòu)下沒有這些問題,在單體架構(gòu)下同樣要避免重復(fù)請求。但是出現(xiàn)的問題要比這少得多。 <>


          為了解決以上問題,就需要保證接口的冪等性,接口的冪等性實際上就是接口可重復(fù)調(diào)用,在調(diào)用方多次調(diào)用的情況下,接口最終得到的結(jié)果是一致的。有些接口可以天然的實現(xiàn)冪等性,比如查詢接口,對于查詢來說,你查詢一次和兩次,對于系統(tǒng)來說,沒有任何影響,查出的結(jié)果也是一樣。

          除了查詢功能具有天然的冪等性之外,增加、更新、刪除都要保證冪等性。那么如何來保證冪等性呢?

          <>全局唯一ID


          如果使用全局唯一ID,就是根據(jù)業(yè)務(wù)的操作和內(nèi)容生成一個全局ID,在執(zhí)行操作前先根據(jù)這個全局唯一ID是否存在,來判斷這個操作是否已經(jīng)執(zhí)行。如果不存在則把全局ID,存儲到存儲系統(tǒng)中,比如數(shù)據(jù)庫、redis等。如果存在則表示該方法已經(jīng)執(zhí)行。


          從工程的角度來說,使用全局ID做冪等可以作為一個業(yè)務(wù)的基礎(chǔ)的微服務(wù)存在,在很多的微服務(wù)中都會用到這樣的服務(wù),在每個微服務(wù)中都完成這樣的功能,會存在工作量重復(fù)。另外打造一個高可靠的冪等服務(wù)還需要考慮很多問題,比如一臺機器雖然把全局ID先寫入了存儲,但是在寫入之后掛了,這就需要引入全局ID的超時機制。


          使用全局唯一ID是一個通用方案,可以支持插入、更新、刪除業(yè)務(wù)操作。但是這個方案看起來很美但是實現(xiàn)起來比較麻煩,下面的方案適用于特定的場景,但是實現(xiàn)起來比較簡單。

          <>去重表


          這種方法適用于在業(yè)務(wù)中有唯一標的插入場景中,比如在以上的支付場景中,如果一個訂單只會支付一次,所以訂單ID可以作為唯一標識。這時,我們就可以建一張去重表,并且把唯一標識作為唯一索引,在我們實現(xiàn)時,把創(chuàng)建支付單據(jù)和寫入去去重表,放在一個事務(wù)中,如果重復(fù)創(chuàng)建,數(shù)據(jù)庫會拋出唯一約束異常,操作就會回滾。

          <>插入或更新


          這種方法插入并且有唯一索引的情況,比如我們要關(guān)聯(lián)商品品類,其中商品的ID和品類的ID可以構(gòu)成唯一索引,并且在數(shù)據(jù)表中也增加了唯一索引。這時就可以使用InsertOrUpdate操作。在mysql數(shù)據(jù)庫中如下:

          1 2 3 4 insert into goods_category
          (goods_id,category_id,create_time,update_time) values(#{goodsId},#{categoryId},
          now(),now()) on DUPLICATE KEY UPDATE update_time=now()
          <>多版本控制

          這種方法適合在更新的場景中,比如我們要更新商品的名字,這時我們就可以在更新的接口中增加一個版本號,來做冪等

          1 boolean updateGoodsName(int id,String newName,int version);
          在實現(xiàn)時可以如下

          1 update goods set name=#{newName},version=#{version} where id=#{id} and
          version<${version}
          <>狀態(tài)機控制


          這種方法適合在有狀態(tài)機流轉(zhuǎn)的情況下,比如就會訂單的創(chuàng)建和付款,訂單的付款肯定是在之前,這時我們可以通過在設(shè)計狀態(tài)字段時,使用int類型,并且通過值類型的大小來做冪等,比如訂單的創(chuàng)建為0,付款成功為100。付款失敗為99

          在做狀態(tài)機更新時,我們就這可以這樣控制

          1 update `order` set status=#{status} where id=#{id} and status<#{status}
          以上就是保證接口冪等性的一些方法。

          友情鏈接
          ioDraw流程圖
          API參考文檔
          OK工具箱
          云服務(wù)器優(yōu)惠
          阿里云優(yōu)惠券
          騰訊云優(yōu)惠券
          京東云優(yōu)惠券
          站點信息
          問題反饋
          郵箱:[email protected]
          QQ群:637538335
          關(guān)注微信

                水多多成人网页 | 李思思一级婬片A片 | 国产精品秘 久久久久久奇米影视 | 性处破与摘花过程videos | 精精国产xxxx视频在线播放 |