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

服務(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

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 四虎地址8848aa4hc44四虎 四虎成人永久地址 | 1024亚洲天堂 | 日本xxxⅹ69xxxx护士 | 成人免费网址 | 亚洲国产视频网站 | 亚洲国产一区二区a毛片 | 草莓绿巨人香蕉茄子芭乐 | 性xxx免费视频 | 久久免费资源福利资源站 | 無码一区中文字幕少妇熟女网站 | 无敌在线视频观看免费 | 男人猛戳女人下部30分钟 | 亚洲社区在线观看 | 国产午夜精品久久理论片小说 | 小舞丝袜调教喷水沦为肉奴 | 国产精品污双胞胎在线观看 | 2020韩国三级理论在线观看 | 超碰97 | 高清视频在线播放ww | 国产日韩欧美 | 四色6677最新永久网站 | 久久久久久久电影 | 香蕉97超级碰碰碰免费公 | 免费在线视频观看 | 国产欧美一区二区精品久久久 | 美女的隐私视频免费看软件 | 男女做污事 | xxxx成人| 逼逼日| 亚洲成人一区在线 | 国产毛片在线观看 | 小柔的性放荡羞辱日记动漫 | 美女扒开两腿露出尿口的视频 | 久久三级网站 | japanese秘书丝袜 | 国产精品九九热 | av72成人| 国产成年人视频 | 国产精品久久久久久久牛牛 | 国产免费午夜 | 欧美 国产 日韩 第一页 |