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

Spring Boot2+JPA之悲觀鎖和樂觀鎖實戰(zhàn)教程

 更新時間:2021年10月15日 09:42:28   作者:Moshow鄭鍇  
這篇文章主要介紹了Spring Boot2+JPA之悲觀鎖和樂觀鎖實戰(zhàn)教程,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教

前言

大量的請求,或者同時的操作,容易導致系統(tǒng)在業(yè)務上發(fā)生并發(fā)的問題. 通常講到并發(fā),解決方案無非就是前端限制重復提交,后臺進行悲觀鎖或者樂觀鎖限制.

悲觀鎖與并發(fā)

悲觀鎖(Pessimistic Lock),顧名思義,就是很悲觀,每次去拿數(shù)據(jù)的時候都認為別人會修改,所以每次在拿數(shù)據(jù)的時候都會上鎖,這樣別人想拿這個數(shù)據(jù)就會block直到解鎖,可以理解為獨占鎖。在java中synchronized和ReentrantLock重入鎖等鎖就是悲觀鎖,數(shù)據(jù)庫中表鎖、行鎖、讀寫鎖等也是悲觀鎖。

利用SQL的for update解決并發(fā)問題

行鎖就是操作數(shù)據(jù)的時候把這一行數(shù)據(jù)鎖住,其他線程想要讀寫必須等待,但同一個表的其他數(shù)據(jù)還是能被其他線程操作的。只要在需要查詢的sql后面加上for update,就能鎖住查詢的行,特別要注意查詢條件必須要是索引列,如果不是索引就會變成表鎖,把整個表都鎖住。

public interface ArticleRepository extends JpaRepository<Article, Long> {
    @Query(value = "select * from article a where a.id = :id for update", nativeQuery = true)
    Optional<Article> findArticleForUpdate(Long id);
}

利用JPA的@Lock行鎖注解解決并發(fā)問題

如果說for update的做法太原始,那么JPA有提供一個更加優(yōu)雅的方法,就是@Lock注解 .

為Repository添加JPA的鎖方法,其中LockModeType.PESSIMISTIC_WRITE參數(shù)就是行鎖。

關于LockModeType這個類型,可以在這找到文檔 https://docs.oracle.com/javaee/7/api/javax/persistence/LockModeType.html

  • NONE: No lock.
  • OPTIMISTIC: Optimistic lock.
  • OPTIMISTIC_FORCE_INCREMENT: Optimistic lock, with version update.
  • PESSIMISTIC_FORCE_INCREMENT: Pessimistic write lock, with version update.
  • PESSIMISTIC_READ: Pessimistic read lock.
  • PESSIMISTIC_WRITE: Pessimistic write lock.
  • READ: Synonymous with OPTIMISTIC.
  • WRITE: Synonymous with OPTIMISTIC_FORCE_INCREMENT.
public interface ArticleRepository extends JpaRepository<Article, Long> { 
    @Lock(value = LockModeType.PESSIMISTIC_WRITE)
    @Query("select a from Article a where a.id = :id")
    Optional<Article> findArticleWithPessimisticLock(Long id);
}

如果是@NameQuery,則可以

@NamedQuery(name="lockArticle",query="select a from Article a where a.id = :id",lockMode = PESSIMISTIC_READ)
public class Article

如果用entityManager的方式,則可以設置LocakMode:

 Query query = entityManager.createQuery("from Article where articleId = :id");
 query.setParameter("id", id);
 query.setLockMode(LockModeType.PESSIMISTIC_WRITE);
 query.getResultList();

樂觀鎖與并發(fā)

樂觀鎖(Optimistic Lock),顧名思義,就是很樂觀,每次去拿數(shù)據(jù)的時候都認為別人不會修改,所以不會上鎖,但是在提交更新的時候會判斷一下在此期間別人有沒有去修改。所以悲觀鎖是限制其他線程,而樂觀鎖是限制自己,雖然他的名字有鎖,但是實際上不算上鎖,通常為version版本號機制,還有CAS算法 .

利用version字段解決并發(fā)問題

版本號機制就是在數(shù)據(jù)庫中加一個字段version當作版本號。那么獲取Article的時候就會帶一個版本號,比如version=1,然后你對這個Article一波操作,操作完之后要插入到數(shù)據(jù)庫了。

校驗一下version版本號,發(fā)現(xiàn)在數(shù)據(jù)庫里對應Article記錄的version=2,這和我手里的版本不一樣啊,說明提交的Article不是最新的,那么就不能update到數(shù)據(jù)庫了,進行報錯把,這樣就避免了并發(fā)時數(shù)據(jù)沖突的問題。

public interface ArticleRepository extends JpaRepository<Article, Long> {
    @Modifying
    @Query(value = "update article set content= :content, version = version + 1 where id = :id and version = :version", nativeQuery = true)
    int updateArticleWithVersion(Long id, String content, Long version);
}
public void postComment(Long articleId, String content) {
 //get article
    Optional<Article> articleOptional = articleRepository.findById(articleId);
    //update with Optimistic Lock
    int count = articleRepository.updateArticleWithVersion(article.getId(), content, article.getVersion());
  
    if (count == 0) {
        throw new RuntimeException("更新數(shù)據(jù)失敗,請刷新重試");
    }else{
     articleRepository.save(article);
    } 
}

利用JPA的@Version版本機制解決并發(fā)問題

有沒有更優(yōu)雅的方式? 當然,必須有,那就是JPA自帶的@Version方式實現(xiàn)樂觀鎖。

  • each entity class must have only one version attribute .每個實體類只能有一個@Version字段,不能多
  • it must be placed in the primary table for an entity mapped to several tables . 對于映射到多個表的實體,必須將其放置在主表中
  • type of a version attribute must be one of the following: int, Integer, long, Long, short, Short, java.sql.Timestamp ,

@Version支持的類型必須是以下類型:

  • int
  • Integer
  • long
  • Long
  • short
  • Short
  • java.sql.Timestamp

首先在Article實體類的version字段上加上@Version注解

@Data
@Entity
public class Article{ 
    @Id
    private Long id;  
  //......  
    @Version
    private Integer version; 
}
Article article = entityManager.find(Article.class, id);
entityManager.lock(article , LockModeType.OPTIMISTIC);
entityManager.refresh(article , LockModeType.READ);

什么時候用悲觀鎖或者樂觀鎖

悲觀鎖適合寫多讀少的場景。因為在使用的時候該線程會獨占這個資源,就適合用悲觀鎖,否則用戶只是瀏覽文章的話,用悲觀鎖就會經(jīng)常加鎖,增加了加鎖解鎖的資源消耗。

樂觀鎖適合寫少讀多的場景。由于樂觀鎖在發(fā)生沖突的時候會回滾或者重試,如果寫的請求量很大的話,就經(jīng)常發(fā)生沖突,結(jié)合事務會有經(jīng)常的回滾和重試,這樣對系統(tǒng)資源消耗也是非常大。

所以悲觀鎖和樂觀鎖沒有絕對的好壞,必須結(jié)合具體的業(yè)務情況來決定使用哪一種方式。另外在阿里巴巴開發(fā)手冊里也有提到:

如果每次訪問沖突概率小于 20%,推薦使用樂觀鎖,否則使用悲觀鎖。樂觀鎖的重試次數(shù)不得小于3次。

阿里巴巴建議以沖突概率20%這個數(shù)值作為分界線來決定使用樂觀鎖和悲觀鎖,雖然說這個數(shù)值不是絕對的,但是作為阿里巴巴各個大佬總結(jié)出來的也是一個很好的參考。

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關文章

  • Spring容器注入bean的幾種方式詳解

    Spring容器注入bean的幾種方式詳解

    這篇文章主要介紹了Spring容器注入bean的幾種方式詳解,@Configuration用來聲明一個配置類,然后使用 @Bean 注解,用于聲明一個bean,將其加入到Spring容器中,這種方式是我們最常用的一種,需要的朋友可以參考下
    2024-01-01
  • mybatis實現(xiàn)獲取入?yún)⑹荓ist和Map的取值

    mybatis實現(xiàn)獲取入?yún)⑹荓ist和Map的取值

    這篇文章主要介紹了mybatis實現(xiàn)獲取入?yún)⑹荓ist和Map的取值問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-06-06
  • Java實體類實現(xiàn)鏈式操作實例解析

    Java實體類實現(xiàn)鏈式操作實例解析

    這篇文章主要介紹了Java實體類實現(xiàn)鏈式操作實例解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-12-12
  • Springboot詳解底層啟動過程

    Springboot詳解底層啟動過程

    這篇文章主要介紹了SpringBoot啟動過程的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-07-07
  • uploadify上傳及后臺文件合法性驗證的代碼解析

    uploadify上傳及后臺文件合法性驗證的代碼解析

    這篇文章主要介紹了uploadify上傳及后臺文件合法性驗證的代碼解析,整段代碼分為后臺上傳方法,文件合法性驗證類,前端上傳js,非常不錯具有參考借鑒價值,需要的朋友可以參考下
    2016-11-11
  • Java線程監(jiān)聽,意外退出線程后自動重啟的實現(xiàn)方法

    Java線程監(jiān)聽,意外退出線程后自動重啟的實現(xiàn)方法

    下面小編就為大家?guī)硪黄狫ava線程監(jiān)聽,意外退出線程后自動重啟的實現(xiàn)方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-03-03
  • 淺談Spring bean 生命周期驗證

    淺談Spring bean 生命周期驗證

    本篇文章主要介紹了淺談Spring bean 生命周期驗證,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-10-10
  • Java并發(fā)教程之Callable和Future接口詳解

    Java并發(fā)教程之Callable和Future接口詳解

    Java從發(fā)布的第一個版本開始就可以很方便地編寫多線程的應用程序,并在設計中引入異步處理,這篇文章主要給大家介紹了關于Java并發(fā)教程之Callable和Future接口的相關資料,需要的朋友可以參考下
    2021-07-07
  • java如何利用poi解析doc和docx中的數(shù)據(jù)

    java如何利用poi解析doc和docx中的數(shù)據(jù)

    這篇文章主要給大家介紹了關于java如何利用poi解析doc和docx中數(shù)據(jù)的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-04-04
  • 淺談Spring Boot 開發(fā)REST接口最佳實踐

    淺談Spring Boot 開發(fā)REST接口最佳實踐

    這篇文章主要介紹了淺談Spring Boot 開發(fā)REST接口最佳實踐,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-01-01

最新評論