亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java工作中常見的并發(fā)問題處理方法總結

 更新時間:2021年02月10日 08:41:58   作者:Dreamer-1  
這篇文章主要介紹了Java工作中常見的并發(fā)問題處理方法總結,文章內容講解的很清晰,有不太懂得同學可以跟著學習下

問題復現(xiàn)

1. “設備Aの奇怪分身”

時間回到很久很久以前的一個深夜,那時我開發(fā)的多媒體廣告播放控制系統(tǒng)剛剛投產(chǎn)上線,公司開出的第一家線下生鮮店里,幾十個大大小小的多媒體硬件設備正常聯(lián)網(wǎng)后,正由我一臺一臺的注冊及接入到已經(jīng)上線的多媒體廣告播控系統(tǒng)中。
注冊過程簡述如下:

每一個設備注冊到系統(tǒng)中后,相應的在數(shù)據(jù)庫設備表中都會新增一條記錄,來存儲這個設備的各項信息。
本來一切都有條不紊的進行著,直到設備A的注冊打破了這默契的寧靜……
設備A注冊完成后,我突然發(fā)現(xiàn),數(shù)據(jù)庫設備表中,新增了兩條記錄,而且是兩條一模一樣的記錄!
我開始以為自己眼花了……
仔細一看,確確實實是新增了兩條,而且連設備唯一標識(劃橫線,后面要考)和創(chuàng)建時間都一模一樣!
看著屏幕,我陷入了沉思……
為什么會有兩條呢?
在我的注冊邏輯里,落庫之前會先查一遍數(shù)據(jù)庫該設備是否已存在,如果存在就更新已有的,不存在才新增。
所以我百思不得其解,按這個邏輯,第二條一模一樣的數(shù)據(jù)是哪來的?

2. 真相背后的并發(fā)請求

經(jīng)過一番排查及思考,我發(fā)現(xiàn)問題可能就出在注冊請求上。
設備A在向云端發(fā)送http注冊請求時,可能會同時發(fā)送多個相同請求。
云服務器當時部署在多臺Docker容器上,通過查看日志發(fā)現(xiàn),有兩臺容器同時收到了來自設備A的注冊請求。
由此,我推測:
設備A同時發(fā)送了兩個注冊請求,這兩個請求分別在同一時間打到了云端的不同容器上,按照我的注冊邏輯,這兩個容器接收到注冊請求后,同時去查詢了數(shù)據(jù)庫的設備表,這時候設備表里還沒有設備A的記錄,所以兩臺容器都執(zhí)行了新增的操作,因為速度很快,所以這兩條新增記錄在精確到秒的創(chuàng)建時間上,并沒有體現(xiàn)出差別。

3. 并發(fā)新增的延伸

既然并發(fā)的新增操作會產(chǎn)生問題,那么并發(fā)的更新操作是否會有問題呢?

解決方法

解決并發(fā)新增

1. 數(shù)據(jù)庫唯一索引(UNIQUE INDEX)

在數(shù)據(jù)庫建表的時候,通過對具有唯一性的字段(比如上述的設備唯一標識)創(chuàng)建唯一索引,或對組合起來后就具備唯一性的幾個字段創(chuàng)建聯(lián)合唯一索引。

這樣在并發(fā)新增時,只要有一個新增成功,其他的新增操作都會因為數(shù)據(jù)庫拋出的異常(java.sql.SQLIntegrityConstraintViolationException)而失敗,我們只需要處理好新增失敗的情況就行了。

注意唯一索引的字段需要非空,因為字段值為空時會導致唯一索引約束失效

2. java分布式鎖

通過在程序中引入分布式鎖,在進行新增操作前需要先獲取分布式鎖,獲取成功才能繼續(xù),否則新增失敗。

這樣也能解決并發(fā)插入帶來的數(shù)據(jù)重復問題,只是引入分布式鎖的同時也增加了系統(tǒng)的復雜性,如果要落庫的數(shù)據(jù)上有唯一性字段的話,還是推薦采用唯一索引的方法。

在構建分布式鎖的過程中,我們需要用到Redis,這里以設備注冊時使用的分布式鎖為例。

分布式鎖簡單問答:

Q:鎖究竟是什么?

A:鎖實質上是存儲在Redis中,基于特定規(guī)則生成的一個字符串(示例里是固定前綴+設備唯一標識),相當于每個設備注冊的時候都有自己對應的一把鎖,因為鎖只有一把,即使該設備有多個相同的注冊請求同時到來,也只有其中獲取到那把鎖的那一個請求能成功走下去。

Q:什么是獲取鎖?

A:同一個設備,基于相同的規(guī)則生成的字符串(后文以Key代稱該字符串)總是相同的,在執(zhí)行新增操作前,先去Redis中查詢這個Key是否存在,如果已存在,就意味著獲取鎖失??;如果不存在,就將這個Key現(xiàn)存到Redis中,如果存儲成功,表示獲取鎖成功,如果存儲失敗,還是意味著獲取鎖失敗。

Q:鎖是怎么工作的?

A:前面說過,同一個設備,基于相同的規(guī)則生成的字符串(Key)總是相同的,在當前線程執(zhí)行新增操作前,先在Redis中查詢這個Key是否存在,如果已存在,表示此時已經(jīng)有別的線程成功獲取了鎖,正在做當前線程想要做的新增操作,則當前線程不需要進行后續(xù)操作了(是的,你是多余的)

當這個Key不存在時,表示現(xiàn)在還沒有其他線程獲得鎖,則當前線程可以繼續(xù)進行下一步操作——在Redis中趕緊存入這個Key,當這個Key存儲失敗時,意味著有別的線程搶先存入了Key成功獲取了鎖,當前線程晚了一步,想做的工作被別人搶先做了(當前線程可以退下了)

當且僅當在Redis中存入這個Key也成功時,表示當前線程終于獲取鎖成功,可以安心進行后面的新增操作了,期間別的想做相同新增操作的線程因為獲取不到鎖,只能全都退場拜拜👋,當前線程執(zhí)行完后要記得釋放鎖(從Redis中刪除這個Key)。

注冊時使用的分布式鎖代碼如下:

public class LockUtil {

 // 對redis底層set/get方法進行了簡單封裝的工具類
 @Autowired
 private RedisService redisService;

 // 生成鎖的固定前綴,從配置文件讀取值
 @Value("${redis.register.prefix}")
 private String REDIS_REGISTER_KEY_PREFIX;

 // 鎖過期時間:即獲取鎖后線程能進行操作的最長時間,超過該時間后鎖自動被釋放(失效),別人可以重新開始獲取鎖進行對應操作
 // 設定鎖過期時間是為了防止某線程成功獲取鎖后在執(zhí)行任務過程中發(fā)生意外掛掉了造成鎖永遠無法被釋放
 @Value("${redis.register.timeout}")
 private Long REDIS_REGISTER_TIMEOUT;

 /**
 * 獲取設備注冊時的分布式鎖
 * @param deviceMacAddress 設備的Mac地址
 * @return
 */
 public boolean getRegisterLock(String deviceMacAddress) {
 if (StringUtils.isEmpty(deviceMacAddress)) {
  return false;
 }

 // 獲取設備對應鎖的字符串(Key)
 String redisKey = getRegisterLockKey(deviceMacAddress);

 // 開始嘗試獲取鎖
 // 如果當前任務鎖key已存在,則表示當前時間內有其他線程正在對該設備執(zhí)行任務,當前線程可以退下了
 if (redisService.exists(redisKey)){
  return false;
 }

 // 開始嘗試加鎖,注意此處需使用SETNX指令(因為可能存在多個線程同時到達這一步開始加鎖,使用SETNX來確保有且僅有一個設置成功返回)
 boolean setLock = redisService.setNX(redisKey, null);

 // 開始嘗試設置鎖過期時間,到了過期時間線程還沒有釋放鎖的話,由保存鎖的Redis來確保鎖最終被釋放,以免出現(xiàn)死鎖
 // 鎖過期時間的設置上,可以評估線程執(zhí)行任務的正常用時,在正常用時的基礎上稍微再大一點
 boolean setExpire = redisService.expire(redisKey, REDIS_REGISTER_TIMEOUT);

 // 設置鎖和設置過期時間均成功時才認為當前線程獲取鎖成功,否則認為獲取鎖失敗
 if (setLock && setExpire) {
  return true;
 }

 // 當發(fā)生設置鎖成功,但設置過期時間失敗的情況時,手動清除剛剛設置的鎖Key
 redisService.del(redisKey);
 return false;
 }

 /**
 * 刪除設備注冊時的分布式鎖
 * @param deviceMacAddress 設備的Mac地址
 */
 public void delRegisterLock(String deviceMacAddress) {
 redisService.del(getRegisterLockKey(deviceMacAddress));
 }

 /**
 * 獲取設備注冊時分布式鎖的key
 * @param deviceMacAddress 設備mac地址(每個設備的mac地址都是唯一的)
 * @return
 */
 private String getRegisterLockKey(String deviceMacAddress) {
 return REDIS_REGISTER_KEY_PREFIX + "_" + deviceMacAddress;
 }
}

在正常的注冊邏輯中使用鎖的示例如下:

public ReturnObj registry(@RequestBody String device){
 Devices deviceInfo = JSON.parseObject(device, Devices.class);

 // 開始注冊前加鎖
 boolean registerLock = lockUtil.getRegisterLock(deviceInfo.getMacAddress());
 if (!registerLock) {
  log.info("獲取設備注冊鎖失敗,當前注冊請求失敗!");
  return ReturnObj.createBussinessErrorResult();
 }

 // 加鎖成功,開始注冊設備
 ReturnObj result = registerDevice(deviceInfo);

 // 注冊設備完成,刪除鎖
 lockUtil.delRegisterLock(deviceInfo.getMacAddress());

 return result;
 }

解決并發(fā)更新

1. 并發(fā)更新真的會引發(fā)問題嗎?

當發(fā)生同時更新或一前一后更新的情況對業(yè)務并無影響的時候,那就無需進行任何處理,免得徒勞增加系統(tǒng)復雜度。

2. 樂觀鎖

通過樂觀鎖的方式可以避免重復更新,即:在數(shù)據(jù)庫表中加入一個“版本號”(version)的字段,在做更新操作前先查詢記錄,記下查詢出的版本號,之后在實際更新操作的時候判斷此前查詢出的版本號是否與當前數(shù)據(jù)庫中該條記錄的版本號一致,如果一致,說明在當前線程從查詢到更新這段時間里,沒有其他線程更新這條記錄;如果不一致,說明再此期間已經(jīng)有其他線程更改了這條記錄,當前線程的更新操作已經(jīng)不安全了,只能放棄。

判斷SQL示例:

update a_table set name=test1, age=12, version=version+1 where id = 3 and version = 1

樂觀鎖通過版本號的方式,在最后更新的關頭才判斷自己之前從數(shù)據(jù)庫讀取的數(shù)據(jù)有沒有被別人修改,其效率高于悲觀鎖,因為在當前線程查詢和最后更新前的這段時間里,其他線程可以照常讀取這同一條記錄,且可以搶先更新。

悲觀鎖

悲觀鎖與樂觀鎖恰好相反,在當前線程查詢這條待更新的數(shù)據(jù)時,就鎖住了這條數(shù)據(jù),不允許在自己更新完成前有其他線程修改數(shù)據(jù)。

通過使用 select … for update 來告訴數(shù)據(jù)庫“我馬上要更新這條數(shù)據(jù),把它給我鎖起來”。

注意:FOR UPDATE 僅適用于InnoDB,且必須在事務中才能生效,當查詢條件有明確主鍵且有此記錄時為行鎖定(row lock,只鎖定根據(jù)查詢條件定位到的這一行數(shù)據(jù)),查詢條件無主鍵或主鍵不明確時為表鎖定(table lock,鎖定全表,會造成全表的數(shù)據(jù)在鎖定期都無法被更改),所以使用悲觀鎖時查詢條件最好能明確定位到某一行或幾行,不要引發(fā)全表鎖定

到此這篇關于Java工作中常見的并發(fā)問題處理方法總結的文章就介紹到這了,更多相關Java工作中并發(fā)問題內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • springboot+vue+elementsUI實現(xiàn)分角色注冊登錄界面功能

    springboot+vue+elementsUI實現(xiàn)分角色注冊登錄界面功能

    這篇文章主要給大家介紹了關于springboot+vue+elementsUI實現(xiàn)分角色注冊登錄界面功能的相關資料,Spring?Boot和Vue.js是兩個非常流行的開源框架,可以用來構建Web應用程序,需要的朋友可以參考下
    2023-07-07
  • 詳解Mybatis動態(tài)sql

    詳解Mybatis動態(tài)sql

    MyBatis的動態(tài)SQL是基于OGNL表達式的,它可以幫助我們方便的在SQL語句中實現(xiàn)某些邏輯。本文給大家介紹Mybatis動態(tài)sql小結,感興趣的朋友參考下
    2016-04-04
  • Spring?Boot?詳細分析Conditional自動化配置注解

    Spring?Boot?詳細分析Conditional自動化配置注解

    首先我們先了解一下@Conditional注解,@Conditional是Spring4新提供的注解,它的作用是按照一定的條件進行判斷,需要注入的Bean滿足給定條件才可以注入到Spring?IOC容器中
    2022-07-07
  • Java中的set集合是什么意思

    Java中的set集合是什么意思

    這篇文章主要介紹了Java中的set集合是什么意思,詳細地講解一下Collection集合中的另外一個分支——Set系列集合,需要的朋友可以參考下
    2022-05-05
  • Java 操作Properties配置文件詳解

    Java 操作Properties配置文件詳解

    本篇文章主要介紹了Java 操作Properties配置文件詳解,詳細的介紹了Properties和主要方法,有興趣的可以了解下
    2017-05-05
  • Java中的IO流原理和流的分類詳解

    Java中的IO流原理和流的分類詳解

    這篇文章主要介紹了Java中的IO流原理和流的分類詳解,Java?io流是Java編程語言中用于輸入和輸出操作的一種機制。它提供了一組類和接口,用于處理不同類型的數(shù)據(jù)流,包括文件、網(wǎng)絡連接、內存等,需要的朋友可以參考下
    2023-10-10
  • mybatis升級mybatis-plus時踩到的一些坑

    mybatis升級mybatis-plus時踩到的一些坑

    這篇文章主要給大家介紹了關于mybatis升級mybatis-plus時踩到的一些坑,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-09-09
  • 淺談SpringMVC請求映射handler源碼解讀

    淺談SpringMVC請求映射handler源碼解讀

    這篇文章主要介紹了淺談SpringMVC請求映射handler源碼解讀,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-03-03
  • java的url方式、本地方式獲取json文件內容

    java的url方式、本地方式獲取json文件內容

    這篇文章給大家分享了java的url方式、本地方式獲取json文件內容的實例代碼,有需要的朋友參考學習下。
    2018-07-07
  • Java多線程yield心得分享

    Java多線程yield心得分享

    前幾天復習了一下多線程,發(fā)現(xiàn)有許多網(wǎng)上講的都很抽象,所以,自己把網(wǎng)上的一些案例總結了一下
    2013-12-12

最新評論