# Hibernate 4 的并發控制
> 原文: [https://javabeginnerstutorial.com/hibernate/concurrency-control-with-hibernate-4/](https://javabeginnerstutorial.com/hibernate/concurrency-control-with-hibernate-4/)
在大多數情況下,僅讓數據庫執行并發控制工作是可以的,但是有時您會遇到需要接管的應用。
在本文中,我將簡要介紹**樂觀**和**悲觀**并發控制。
## 樂觀并發控制
高并發應用中唯一一致的方法是帶有版本控制的樂觀并發控制。 此方法使用版本號或時間戳來檢測沖突并防止更新丟失。
### 應用版本檢查
使用這種方法,應用必須手動維護實體的版本。 這意味著開發人員有責任在操作它們之前從數據庫加載實際的實體狀態。 當對象被 Hibernate 刷新時,版本會自動增加-因此開發人員不需要增加此屬性。
如果您的應用具有低并發性并跳過版本控制,則可以使用此方法。 在這種情況下,總是最后一次提交獲勝,并且根據該狀態更新對象。 這就是為什么**總是**需要在操作數據庫之前先從數據庫中加載實體的實際狀態。
### 自動版本控制
要使用自動版本控制,只需在您希望在樂觀鎖定的版本控制下擁有的實體中添加一個字段或方法,然后使用 [[受電子郵件保護]](/cdn-cgi/l/email-protection) 注解對其進行注解。 如文檔所述,此注解可用于以下類型:`int`,`Integer`,`short`,`Short`,`long`,`Long`,`java.sql.Timestamp`。
如果在加載實體和將其刷新回數據庫之間發生一些更新,則會從 Hibernate 收到一條錯誤消息:
線程“主”中的異常`org.hibernate.StaleObjectStateException`:行已由另一個事務更新或刪除(或未保存值的映射不正確):`[hibernate_example.joined.Book#1]`
## 悲觀并發控制
正如我上次已經提到的那樣,Hibernate 不會鎖定內存中的對象,它將始終使用基礎數據庫的鎖定機制。
但是, `LockMode`類定義了一些可由 Hibernate 獲得的機制:
* `WRITE`:在 Hibernate 更新或插入行時自動獲取
* `UPGRADE`:可以在明確的用戶請求下使用`SELECT…FOR UPDATE`在支持此語法的數據庫上獲取
* `UPGRADE_NOWAIT`:可在 Oracle 下使用`SELECT…FOR UPDATE NOWAIT`根據明確的用戶請求獲取
* `READ`:Hibernate 讀取數據時自動獲取
* `NONE`:表示沒有鎖,在事務結束時所有對象都切換到此鎖模式
* `PESSIMISTIC_FORCE_INCREMENT`:加載實體時強制增加版本。
上面提到的“顯式用戶請求”可以表示為以下調用之一:
* 為`LockOptions`參數指定`LockMode`的`load()`
* `buildLockRequest()`
* `setLockMode()`
## 設置鎖定模式的示例
上面我已經提到了一些鎖定模式以及如何設置它們,現在是時候看看一些示例代碼來了解它們的作用了。
```java
getBooks(session).stream().forEach(b -> session.load(Book.class, b.getId(),
new LockOptions(LockMode.PESSIMISTIC_FORCE_INCREMENT)));
getBooks(session).stream().forEach(System.out::println);
```
上面的代碼塊有兩件事:從數據庫加載實體,設置一種用于強制版本增加的鎖定模式,然后將書籍打印到控制臺。 結果將如下所示:
```java
Java 8 in Action by Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft (ISBN: 9781617291999), published 2015.07.29\. 0:00 [1]
Java 8 in Action by Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft (ISBN: 9781617291999), published 2015.07.29\. 0:00 [1]
Java 8 in Action by Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft (ISBN: 9781617291999), published 2015.07.29\. 0:00 [1]
Java 8 in Action by Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft (ISBN: 9781617291999), published 2015.07.29\. 16:14 [1]
```
每行末尾的方括號包含實體的版本號。
如果我們想將鎖定移動到`getBooks()`方法中,可以執行以下操作:
```java
private static List<Book> getBooks(Session session) {
?? final Query query = session.createQuery("from Book b");
?? query.setLockMode("b", LockMode.PESSIMISTIC_FORCE_INCREMENT);
?? return query.list();
}
```
有趣的是`setLockMode`方法的第一個字符串參數:它是實體使用此鎖定的別名,您必須在查詢的`FROM`塊中使用此別名。
使用此倉庫方法加載書籍后,將書籍打印到控制臺的結果可能是這樣的:
```java
Java 8 in Action by Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft (ISBN: 9781617291999), published 2015.07.29\. 0:00 [1]
Java 8 in Action by Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft (ISBN: 9781617291999), published 2015.07.29\. 0:00 [1]
Java 8 in Action by Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft (ISBN: 9781617291999), published 2015.07.29\. 0:00 [1]
Java 8 in Action by Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft (ISBN: 9781617291999), published 2015.07.29\. 16:15 [1]
```
### 一個有趣的事實
版本號僅在當前會話中更新,直到您更新實體并將其保存到數據庫為止。
如果不這樣做,您會在應用中看到版本增量,但是它們不會保存到數據庫中,即使您使用某些`*_FORCE_INCREMENT`策略。
```java
| 1 |?? Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft | 9781617291999 | 2015-07-29 | Java 8 in Action | 0 |
| 2 | Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft | 9781617291999 |?? 2015-07-29 | Java 8 in Action | 0 |
| 3 | Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft | 9781617291999 |?? 2015-07-29 | Java 8 in Action | 0 |
| 4 | Raoul-Gabriel Urma, Mario Fusco, Alan Mycroft | 9781617291999 |?? 2015-07-29 | Java 8 in Action | 0 |
```
如果我們不更新舊條目,則上面的塊顯示了數據庫中的實體。 這些列從左到右如下:ID,作者,ISBN,出版,標題和版本。
要進行一些版本更改,請取消注釋示例中的代碼塊,然后運行應用。 該塊將書籍的日期更新為當前日期,并將實體保存回數據庫中。 為此,您需要進行事務。
要開始事務,只需調用`session.beginTransaction();`,如果完成調用`session.getTransaction().commite()`, 完成并寫入數據庫或`session.getTransaction().rollback()`以還原此事務中所做的所有更改。 或者,您可以存儲由`beginTransaction()`方法返回的*事務*對象,并在此*事務*實例上調用`commit()`或`rollback()`。
## 一些事務方式
如果我提到了事務,請允許我給您一些模式(甚至是反模式),以了解通常如何處理事務。
### 每個操作的會話(反模式)
這是一種反模式,因為如果使用此事務管理方法,則將為每個數據庫調用打開和關閉會話。 即使您使用數據庫的*自動提交*功能(在每次調用數據庫后隱式調用*提交*),此操作也會完成。
### 每個請求的會話
這是用于事務的最常見模式。 名稱中的**請求**與其中有來自客戶端/用戶(例如 Web 應用)的許多請求的系統有關。
常見的工作流程是,當此類請求到達系統時,將打開 Hibernate *會話*,并保持打開狀態,直到處理該請求的信息(更新存儲的信息或檢索要顯示的內容)為止。
如果使用此模式,則在大多數情況下可以減少數據丟失,因為信息會連續保存在數據庫中。
### 每個應用的會話
開發人員不同意這是模式還是反模式,因為有時應用很小,您確實可以打開一個事務并在最后調用`commit`。 但是,對于大型應用而言,這是一種太糟糕的方法,在大型應用中,最終的應用故障會導致數據丟失,或者如果存在并發用戶并且數據相關,則這些用戶之間將無法處于同步狀態。
## 總結
鎖定可能會有些麻煩,但是 Hibernate 會通過各種鎖定機制為我們提供幫助和幫助。 如果您想微調內置解決方案,也可以使用上述方法進行。
我們稍微深入研究事務,只是為了了解如何通常使用設計模式進行處理,以了解如何將數據存儲在數據庫中并保留這些信息。 您可以[從此處下載代碼](https://github.com/JBTAdmin/Hibernate)。
- JavaBeginnersTutorial 中文系列教程
- Java 教程
- Java 教程 – 入門
- Java 的歷史
- Java 基礎知識:Java 入門
- jdk vs jre vs jvm
- public static void main(string args[])說明
- 面向初學者的 Java 類和對象教程
- Java 構造器
- 使用 Eclipse 編寫 Hello World 程序
- 執行順序
- Java 中的訪問修飾符
- Java 中的非訪問修飾符
- Java 中的數據類型
- Java 中的算術運算符
- Java 語句初學者教程
- 用 Java 創建對象的不同方法
- 內部類
- 字符串構建器
- Java 字符串教程
- Java 教程 – 變量
- Java 中的變量
- Java 中的局部變量
- Java 中的實例變量
- Java 引用變量
- 變量遮蓋
- Java 教程 – 循環
- Java for循環
- Java 教程 – 異常
- Java 異常教程
- 異常處理 – try-with-resources語句
- Java 異常處理 – try catch塊
- Java 教程 – OOPS 概念
- Java 重載
- Java 方法覆蓋
- Java 接口
- 繼承
- Java 教程 – 關鍵字
- Java 中的this關鍵字
- Java static關鍵字
- Java 教程 – 集合
- Java 數組教程
- Java 集合
- Java 集合迭代器
- Java Hashmap教程
- 鏈表
- Java 初學者List集合教程
- Java 初學者的Map集合教程
- Java 初學者的Set教程
- Java 初學者的SortedSet集合教程
- Java 初學者SortedMap集合教程
- Java 教程 – 序列化
- Java 序列化概念和示例
- Java 序列化概念和示例第二部分
- Java 瞬態與靜態變量
- serialVersionUID的用途是什么
- Java 教程 – 枚舉
- Java 枚舉(enum)
- Java 枚舉示例
- 核心 Java 教程 – 線程
- Java 線程教程
- Java 8 功能
- Java Lambda:初學者指南
- Lambda 表達式簡介
- Java 8 Lambda 列表foreach
- Java 8 Lambda 映射foreach
- Java 9
- Java 9 功能
- Java 10
- Java 10 獨特功能
- 核心 Java 教程 – 高級主題
- Java 虛擬機基礎
- Java 類加載器
- Java 開發人員必須知道..
- Selenium 教程
- 1 什么是 Selenium?
- 2 為什么要進行自動化測試?
- 3 Selenium 的歷史
- 4 Selenium 工具套件
- 5 Selenium 工具支持的瀏覽器和平臺
- 6 Selenium 工具:爭霸
- 7A Selenium IDE – 簡介,優點和局限性
- 7B Selenium IDE – Selenium IDE 和 Firebug 安裝
- 7C Selenium IDE – 突破表面:初探
- 7D Selenium IDE – 了解您的 IDE 功能
- 7E Selenium IDE – 了解您的 IDE 功能(續)。
- 7F Selenium IDE – 命令,目標和值
- 7G Selenium IDE – 記錄和運行測試用例
- 7H Selenium IDE – Selenium 命令一覽
- 7I Selenium IDE – 設置超時,斷點,起點
- 7J Selenium IDE – 調試
- 7K Selenium IDE – 定位元素(按 ID,名稱,鏈接文本)
- 7L Selenium IDE – 定位元素(續)
- 7M Selenium IDE – 斷言和驗證
- 7N Selenium IDE – 利用 Firebug 的優勢
- 7O Selenium IDE – 以所需的語言導出測試用例
- 7P Selenium IDE – 其他功能
- 7Q Selenium IDE – 快速瀏覽插件
- 7Q Selenium IDE – 暫停和反射
- 8 給新手的驚喜
- 9A WebDriver – 架構及其工作方式
- 9B WebDriver – 在 Eclipse 中設置
- 9C WebDriver – 啟動 Firefox 的第一個測試腳本
- 9D WebDriver – 執行測試
- 9E WebDriver – 用于啟動其他瀏覽器的代碼示例
- 9F WebDriver – JUnit 環境設置
- 9G WebDriver – 在 JUnit4 中運行 WebDriver 測試
- 9H WebDriver – 隱式等待
- 9I WebDriver – 顯式等待
- 9J WebDriver – 定位元素:第 1 部分(按 ID,名稱,標簽名稱)
- 9K WebDriver – 定位元素:第 2 部分(按className,linkText,partialLinkText)
- 9L WebDriver – 定位元素:第 3a 部分(按cssSelector定位)
- 9M WebDriver – 定位元素:第 3b 部分(cssSelector續)
- 9N WebDriver – 定位元素:第 4a 部分(通過 xpath)
- 9O WebDriver – 定位元素:第 4b 部分(XPath 續)
- 9P WebDriver – 節省時間的捷徑:定位器驗證
- 9Q WebDriver – 處理驗證碼
- 9R WebDriver – 斷言和驗證
- 9S WebDriver – 處理文本框和圖像
- 9T WebDriver – 處理單選按鈕和復選框
- 9U WebDriver – 通過兩種方式選擇項目(下拉菜單和多項選擇)
- 9V WebDriver – 以兩種方式處理表
- 9W WebDriver – 遍歷表元素
- 9X WebDriver – 處理警報/彈出框
- 9Y WebDriver – 處理多個窗口
- 9Z WebDriver – 最大化窗口
- 9AA WebDriver – 執行 JavaScript 代碼
- 9AB WebDriver – 使用動作類
- 9AC WebDriver – 無法輕松定位元素? 繼續閱讀...
- 10A 高級 WebDriver – 使用 Apache ANT
- 10B 高級 WebDriver – 生成 JUnit 報告
- 10C 高級 WebDriver – JUnit 報表自定義
- 10D 高級 WebDriver – JUnit 報告自定義續
- 10E 高級 WebDriver – 生成 PDF 報告
- 10F 高級 WebDriver – 截屏
- 10G 高級 WebDriver – 將屏幕截圖保存到 Word 文檔
- 10H 高級 WebDriver – 發送帶有附件的電子郵件
- 10I 高級 WebDriver – 使用屬性文件
- 10J 高級 WebDriver – 使用 POI 從 excel 讀取數據
- 10K 高級 WebDriver – 使用 Log4j 第 1 部分
- 10L 高級 WebDriver – 使用 Log4j 第 2 部分
- 10M 高級 WebDriver – 以無頭模式運行測試
- Vue 教程
- 1 使用 Vue.js 的 Hello World
- 2 模板語法和反應式的初探
- 3 Vue 指令簡介
- 4 Vue Devtools 設置
- 5 數據綁定第 1 部分(文本,原始 HTML,JavaScript 表達式)
- 6 數據綁定第 2 部分(屬性)
- 7 條件渲染第 1 部分(v-if,v-else,v-else-if)
- 8 條件渲染第 2 部分(v-if和v-show)
- 9 渲染列表第 1 部分(遍歷數組)
- 10 渲染列表第 2 部分(遍歷對象)
- 11 監聽 DOM 事件和事件修飾符
- 12 監聽鍵盤和鼠標事件
- 13 讓我們使用簡寫
- 14 使用v-model進行雙向數據綁定
- 15 表單輸入綁定
- 18 類綁定
- Python 教程
- Python 3 簡介
- Python 基礎知識 - 又稱 Hello World 以及如何實現
- 如何在 Windows 中安裝 python
- 適用于 Windows,Mac,Linux 的 Python 設置
- Python 數字和字符串
- Python 列表
- Python 集
- Python 字典
- Python 條件語句
- Python 循環
- Python 函數
- 面向對象編程(OOP)
- Python 中的面向對象編程
- Python 3 中的異常處理
- Python 3:猜數字
- Python 3:猜數字 – 回顧
- Python 生成器
- Hibernate 教程
- Hibernate 框架基礎
- Hibernate 4 入門教程
- Hibernate 4 注解配置
- Hibernate 4 的實體關系
- Hibernate 4 中的實體繼承模型
- Hibernate 4 查詢語言
- Hibernate 4 數據庫配置
- Hibernate 4 批處理
- Hibernate 4 緩存
- Hibernate 4 審計
- Hibernate 4 的并發控制
- Hibernate 4 的多租戶
- Hibernate 4 連接池
- Hibernate 自舉