JPA 加鎖機制及@Version版本控制方式
JPA的加鎖機制有兩種,樂觀鎖和悲觀鎖。
樂觀鎖:
樂觀鎖的特點在于認為數(shù)據(jù)沖突或者更新丟失等情況是很少發(fā)生的.當發(fā)生的時候,拋出異常和回滾就足夠解決問題.
悲觀鎖:
悲觀鎖的邏輯在于認為每次數(shù)據(jù)操作都很有可能發(fā)生沖突,所以一開始就獲得記錄的鎖,再進行記錄的操作是解決問題的優(yōu)先選擇.
一 簡述悲觀鎖的用法
悲觀鎖通常是SQL級別的,通過讀寫時先拿到鎖實現(xiàn),在SQL語句中就會有體現(xiàn).
1.1 EntityManager 用法
return em.createQuery(sql 語句).setLockMode(LockModeType.NONE).getResultList();
//分解寫法大概是:
Query query = getSession().createQuery(hql);
query.setLockMode(LockModeType.NONE);
EntityManager 是一個輔助類,createQuery后返回的就是一個Query對象,然后通過
setLockMode設(shè)置鎖的級別即可.
| LockModeType 類型 | 解釋 |
|---|---|
| LockMode.READ | 事務(wù)的隔離級別是Repeatable Read或Serializable時,請求讀取數(shù)據(jù)庫記錄時自動獲得 |
| LockMode.WRITE | 請求插入或更新數(shù)據(jù)庫記錄時自動獲得 |
| 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()、 |
二 樂觀鎖的詳細用法
樂觀鎖本篇的主要內(nèi)容
實體類是關(guān)鍵 , 樂觀鎖常用方法是通過version來控制 ,
- 數(shù)據(jù)庫對應(yīng)的表中需要有一個字段(名字隨意),字段類型設(shè)置成BigInt即可
- 業(yè)務(wù)不對該字段進行控制,字段的控制交由系統(tǒng)處理
- 每一次修改都會導(dǎo)致version遞增
- 當出現(xiàn)同時獲得該記錄的對象且均需要修改時,當?shù)谝粋€已經(jīng)提交事務(wù),version字段發(fā)生改變,后面提交的事務(wù)發(fā)現(xiàn)version版本不對,則無法提交,拋出異常
實體類(注意其中的@Version注解)
@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ù)的提交性
@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
@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是自動增長的,唯一的缺點是拋出的異常不易捕獲,捕獲的方法:
@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,這種是手動的提交方法
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java訪問Hadoop分布式文件系統(tǒng)HDFS的配置說明
Hadoop的能提供高吞吐量的數(shù)據(jù)訪問,是集群式服務(wù)器的上的數(shù)據(jù)操作利器,這里就來為大家分享Java訪問Hadoop分布式文件系統(tǒng)HDFS的配置說明:2016-06-06
SpringTask-Timer實現(xiàn)定時任務(wù)的詳細代碼
在項目中開發(fā)定時任務(wù)應(yīng)該一種比較常見的需求,今天通過示例代碼給大家講解SpringTask-Timer實現(xiàn)定時任務(wù)的相關(guān)知識,感興趣的朋友一起看看吧2024-06-06
springboot整合SSE技術(shù)開發(fā)小結(jié)
本文主要介紹了springboot整合SSE技術(shù)開發(fā)小結(jié),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-11-11
Spring Boot實戰(zhàn)之數(shù)據(jù)庫操作的示例代碼
本篇文章主要介紹了Spring Boot實戰(zhàn)之數(shù)據(jù)庫操作的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-01-01
Spring使用AOP完成統(tǒng)一結(jié)果封裝實例demo
這篇文章主要介紹了Spring使用AOP完成統(tǒng)一結(jié)果封裝,本文通過實現(xiàn)demo給大家詳細講解,代碼簡單易懂,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-02-02
使用SpringBoot創(chuàng)建一個RESTful API的詳細步驟
使用 Java 的 Spring Boot 創(chuàng)建 RESTful API 可以滿足多種開發(fā)場景,它提供了快速開發(fā)、易于配置、可擴展、可維護的優(yōu)點,尤其適合現(xiàn)代軟件開發(fā)的需求,幫助你快速構(gòu)建出高性能的后端服務(wù),需要的朋友可以參考下2025-01-01

