一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Java教程 - JPA 加鎖機(jī)制及@Version版本控制方式

JPA 加鎖機(jī)制及@Version版本控制方式

2022-02-20 12:09black-ant Java教程

這篇文章主要介紹了JPA 加鎖機(jī)制及@Version版本控制方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

JPA的加鎖機(jī)制有兩種,樂觀鎖和悲觀鎖。

樂觀鎖:

樂觀鎖的特點(diǎn)在于認(rèn)為數(shù)據(jù)沖突或者更新丟失等情況是很少發(fā)生的.當(dāng)發(fā)生的時(shí)候,拋出異常和回滾就足夠解決問題.

悲觀鎖:

悲觀鎖的邏輯在于認(rèn)為每次數(shù)據(jù)操作都很有可能發(fā)生沖突,所以一開始就獲得記錄的鎖,再進(jìn)行記錄的操作是解決問題的優(yōu)先選擇.

一 簡述悲觀鎖的用法

悲觀鎖通常是SQL級別的,通過讀寫時(shí)先拿到鎖實(shí)現(xiàn),在SQL語句中就會有體現(xiàn).

1.1 EntityManager 用法

?
1
2
3
4
return em.createQuery(sql 語句).setLockMode(LockModeType.NONE).getResultList();
//分解寫法大概是:
Query query = getSession().createQuery(hql);
query.setLockMode(LockModeType.NONE);

EntityManager 是一個(gè)輔助類,createQuery后返回的就是一個(gè)Query對象,然后通過

setLockMode設(shè)置鎖的級別即可.

LockModeType 類型 解釋
LockMode.READ 事務(wù)的隔離級別是Repeatable Read或Serializable時(shí),請求讀取數(shù)據(jù)庫記錄時(shí)自動(dòng)獲得
LockMode.WRITE 請求插入或更新數(shù)據(jù)庫記錄時(shí)自動(dòng)獲得
LockMode.OPTIMISTIC 樂觀鎖
LockMode.OPTIMISTIC_FORCE_INCREMENT 樂觀鎖,通過version控制
LockMode.PESSIMISTIC_READ 與LockMode.PESSIMISTIC_WRITE相同
LockMode.PESSIMISTIC_WRITE 事務(wù)開始即獲得數(shù)據(jù)庫的鎖
LockMode.PESSIMISTIC_FORCE_INCREMENT 事務(wù)開始即設(shè)置version
LockMode.NONE 取消任何鎖,如事務(wù)結(jié)束后的所有對象,或執(zhí)行了Session的update()、

二 樂觀鎖的詳細(xì)用法

樂觀鎖本篇的主要內(nèi)容

實(shí)體類是關(guān)鍵 , 樂觀鎖常用方法是通過version來控制 ,

  • 數(shù)據(jù)庫對應(yīng)的表中需要有一個(gè)字段(名字隨意),字段類型設(shè)置成BigInt即可
  • 業(yè)務(wù)不對該字段進(jìn)行控制,字段的控制交由系統(tǒng)處理
  • 每一次修改都會導(dǎo)致version遞增
  • 當(dāng)出現(xiàn)同時(shí)獲得該記錄的對象且均需要修改時(shí),當(dāng)?shù)谝粋€(gè)已經(jīng)提交事務(wù),version字段發(fā)生改變,后面提交的事務(wù)發(fā)現(xiàn)version版本不對,則無法提交,拋出異常

實(shí)體類(注意其中的@Version注解)

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
@Entity
public class User {
    @Id
    @GeneratedValue
    private Long id;
    private String username;
    private String userdesc;
    @Version
    private Long version;
    public User() {
    }
    public User(String username, String userdesc) {
        this.username = username;
        this.userdesc = userdesc;
    }
    public Long getId() {
        return id;
    }
    public void setId(Long id) {
        this.id = id;
    }
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String getUserDesc() {
        return userdesc;
    }
    public void setUserDesc(String userdesc) {
        this.userdesc = userdesc;
    }
    public Long getVersion() {
        return version;
    }
    public void setVersion(Long version) {
        this.version = version;
    }
}

controller中通過sleep將線程沉睡,測試事務(wù)的提交性

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
@RestController
public class UserController {
    private Logger logger = LoggerFactory.getLogger(getClass());
    @Autowired
    UserService userService;
    @PostMapping("/changeone")
    @Transactional
    public String changeone() {
        User user = userService.findUser("gang");
        try {
            logger.info("修改1 before:user--{}--Versdion:{}", user.getUserDesc(), user.getVersion());
            Thread.sleep(25000);
            user.setUserDesc("修改1");
            logger.info("修改1 :user--{}--version:{}", user.getUserDesc(), user.getVersion());
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (Exception e) {
            logger.info("eeeeeeeeeeeeee");
            e.printStackTrace();
        }
        return "true";
    }
    @PostMapping("/changetwo")
    @Transactional
    public String changetwo() {
        User user = userService.findUser("gang");
        try {
            logger.info("修改2 before:user--{}--version:{}", user.getUserDesc(), user.getVersion());
            Thread.sleep(30000);
            user.setUserDesc("修改2");
            logger.info("修改2:user--{}--version:{}", user.getUserDesc(), user.getVersion());
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (Exception e) {
            logger.info("eeeeeeeeeeeeee");
            e.printStackTrace();
        }
        return "true";
    }
    @PostMapping("/changethree")
    @Transactional
    public String changethree() {
        User user = userService.findUser("gang");
        logger.info("修改3 before:user--{}--version:{}", user.getUserDesc(), user.getVersion());
        user.setUserDesc("修改3");
        logger.info("修改3 :user--{}--version:{}", user.getUserDesc(), user.getVersion());
        return "true";
    }
    @PostMapping("/newuser")
    @Transactional
    public String newuser() {
        logger.info("save user");
        User user = new User();
        user.setUserDesc("第一次創(chuàng)建");
        user.setUsername("gang");
        userService.saveUser(user);
        return "true";
    }
}

以及service及repository

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@Service
public class UserService {
    @Autowired
    UserRepository userRepository;
    public User findUser(String username){
        return userRepository.findByUsername(username);
    }
    public void saveUser(User user){
        userRepository.save(user);
    }
}
UserRepository
public interface UserRepository extends JpaRepository<User,Long> {
    User findByUsername(String username);
}

總結(jié)

使用很簡單,version是自動(dòng)增長的,唯一的缺點(diǎn)是拋出的異常不易捕獲,捕獲的方法:

?
1
2
3
4
5
6
7
8
9
10
11
12
@Resource
private UserTransaction rtc;
 try {
        rtc.begin();
        User user = userService.findUser("gang");
        user .setDesc("異常捕獲");
         rtc.commit();
    } catch (OptimisticLockException e) {
        throw new OptimisticLockException ();
    } catch (Exception e) {
        throw new Exception ();
    }

注意其中的 rtc.begin(); 以及 rtc.commit();

不同于@Transaction,這種是手動(dòng)的提交方法

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持服務(wù)器之家。

原文鏈接:https://blog.csdn.net/zzg19950824/article/details/85468318

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 欧美精品国产一区二区 | 国产欧美精品一区二区三区 | 精品国产免费第一区二区 | 免费一级毛片完整版在线看 | 国产精品一级香蕉一区 | 五月天精品视频在线观看 | 91制片厂制作传媒网站 | 9999热视频 | 7mav视频 | 欧洲美女人牲交一级毛片 | 天天色国产 | 91麻豆国产精品91久久久 | 久久婷婷五月综合色精品首页 | 成功精品影院 | 性欧洲女人18 | 奇米影视久久777中文字幕 | 亚洲福利电影一区二区? | 亚洲国产精品成人午夜在线观看 | 俺来操 | 男人jj视频 | 农村老少伦小说 | 国产成人在线视频 | 亚洲精品视频观看 | 天堂俺去俺来也www久久婷婷 | 操人网| 9191免费永久观看 | 亚洲精品6久久久久中文字幕 | 久久伊人中文字幕有码 | 国产欧美视频一区二区三区 | 青青草视频破解版 | 欧美一级片观看 | 国语自产拍在线播放不卡 | 国产亚洲精品福利在线 | 日韩色在线观看 | 8天堂资源在线官网 | 美女张开下身让男人桶 | 3d肉浦团在线观看 | 欧美视频一区二区三区四区 | 国产欧美成人免费观看 | 香港三级系列在线播放 | 亚欧洲乱码专区视频 |