1 Hibernate中的事務與併發
1.1 事務相關的概念
1. 什麼是事務* 事務就是邏輯上的一組操作,組成事務的各個執行單元,操作要麼全都成功,要麼全都失敗.
* 轉賬的例子:冠希給美美轉錢,扣錢,加錢。兩個操作組成了一個事情!
2. 事務的特性
* 原子性 -- 事務不可分割.
* 一致性 -- 事務執行的前後數據的完整性保持一致.
* 隔離性 -- 一個事務執行的過程中,不應該受到其他的事務的干擾.
* 持久性 -- 事務一旦提交,數據就永久保持到數據庫中.
3. 如果不考慮隔離性:引發一些讀的問題
* 髒讀 -- 一個事務讀到了另一個事務未提交的數據.
* 不可重複讀 -- 一個事務讀到了另一個事務已經提交的update數據,導致多次查詢結果不一致.
* 虛讀 -- 一個事務讀到了另一個事務已經提交的insert數據,導致多次查詢結果不一致.
4. 通過設置數據庫的隔離級別來解決上述讀的問題
* 未提交讀:以上的讀的問題都有可能發生.
* 已提交讀:避免髒讀,但是不可重複讀,虛讀都有可能發生.
* 可重複讀:避免髒讀,不可重複讀.但是虛讀是有可能發生.
* 串行化:以上讀的情況都可以避免.
5. 如果想在Hibernate的框架中來設置隔離級別,需要在hibernate.cfg.xml的配置文件中通過標籤來配置
* 通過:hibernate.connection.isolation = 4 來配置
* 取值
* 1—Read uncommitted isolation
* 2—Read committed isolation
* 4—Repeatable read isolation
* 8—Serializable isolation
1.2 丟失更新的問題
1. 如果不考慮隔離性,也會產生寫入數據的問題,這一類的問題叫丟失更新的問題。
2. 例如:兩個事務同時對某一條記錄做修改,就會引發丟失更新的問題。
* A事務和B事務同時獲取到一條數據,同時再做修改
* 如果A事務修改完成後,提交了事務
* B事務修改完成後,不管是提交還是回滾,如果不做處理,都會對數據產生影響
3. 解決方案有兩種
* 悲觀鎖
* 採用的是數據庫提供的一種鎖機制,如果採用做了這種機制,在SQL語句的後面添加 for update 子句
* 當A事務在操作該條記錄時,會把該條記錄鎖起來,其他事務是不能操作這條記錄的。
* 只有當A事務提交後,鎖釋放了,其他事務才能操作該條記錄
* 樂觀鎖
* 採用版本號的機制來解決的。會給表結構添加一個字段version=0,默認值是0
* 當A事務在操作完該條記錄,提交事務時,會先檢查版本號,如果發生版本號的值相同時,纔可以提交事務。同時會更新版本號version=1.
* 當B事務操作完該條記錄時,提交事務時,會先檢查版本號,如果發現版本不同時,程序會出現錯誤。
4. 使用Hibernate框架解決丟失更新的問題
* 悲觀鎖
* 使用session.get(Customer.class, 1,LockMode.UPGRADE); 方法
* 樂觀鎖
* 1.在對應的JavaBean中添加一個屬性,名稱可以是任意的。例如:private Integer version; 提供get和set方法
* 2.在映射的配置文件中,提供<version name="version"/>標籤即可。
//解決“讀”問題:設置事務隔離級別
//解決“寫”問題:悲觀鎖 樂觀鎖
1.3 綁定本地的Session
* 一種是通過參數的方式傳遞下去
* 另一種是把Connection綁定到ThreadLocal對象中
2. 現在的Hibernate框架中,使用session對象開啓事務,所以需要來傳遞session對象,框架提供了ThreadLocal的方式
* 需要在hibernate.cfg.xml的配置文件中提供配置
* <property name="hibernate.current_session_context_class">thread</property>
* 重寫HibernateUtil的工具類,使用SessionFactory的getCurrentSession()方法,獲取當前的Session對象。並且該Session對象不用手動關閉,線程結束了,會自動關閉。
public static Session getCurrentSession(){
return factory.getCurrentSession();
}
* 注意:想使用getCurrentSession()方法,必須要先配置才能使用。
這裏有必要說明:一般不會在dao層使用transaction,事務被配置在service層上更爲合理,因爲業務層方法表示邏輯上的一個原子操作。因此,我們應該在Dao層獲取sesssion,然後進行事務處理。
2 Hibernate框架的查詢方式
2.1 Query的查詢接口
Query執行HQL語句:
1.查詢所有記錄
Query query = session.createQuery("from Customer");
List<Customer> list = query.list();
for (Customer customer : list){
System.out.println(customer);
}
2.條件查詢:
(1)
Query query = session.createQuery("from Customer where age > ?");
query.setInteger(0, 18);
List<Customer> list = query.list();
for (Customer customer : list){
System.out.println(customer);
}
也可以指定名稱:
Query query = session.createQuery("from Customer where age > :aaa");
query.setInteger("aaa", 18);
List<Customer> list = query.list();
for (Customer customer : list){
System.out.println(customer);
}
(2)
Query query = session.createQuery("from Customer wherename = ?");
query.setString(0, "小霏");
List<Customer> list = query.list();
for (Customer customer : list){
System.out.println(customer);
}
(3)
Query query = session.createQuery("from Customer where name like ?");
query.setString(0, "%小霞%");
List<Customer> list = query.list();
for (Customer customer : list){
System.out.println(customer);
}
3.條件查詢:
Query query = session.createQuery("from Customer wherename = :aaa and age = :bbb");
query.setString("aaa", "李健");
query.setInteger("bbb", 38);
List<Customer> list = query.list();
for (Customer customer : list){
System.out.println(customer);
}
4.分頁查詢: //MySQL中的LIMIT
Query query = session.createQuery("from Customer");
query.setFirstResult(3);
query.setMaxResults(3);
List<Customer> list = query.list();
for (Customer customer : list){
System.out.println(customer);
}
2.2 Criteria查詢接口(做條件查詢非常合適)
Criteria:用來執行條件查詢
1.查詢所有記錄
//先獲取到Criteria接口
Criteria criteria = session.createCriteria(Customer.class);
List<Customer> list = criteria.list();
for (Customer customer : list){
System.out.println(customer);
}
2.條件查詢
Criteria criteria = session.createCriteria(Customer.class);
//criteria是hibernate提供的條件查詢的對象,想傳入條件得使用工具類Restrictions,裏邊都是靜態方法
criteria.add(Restrictions.eq("name", "小霞"));
//criteria.add(Restrictions.gt("age", 18));
List<Customer> list = criteria.list();
for (Customer customer : list){
System.out.println(customer);
}
3.條件查詢
Criteria criteria = session.createCriteria(Customer.class);
criteria.add(Restrictions.like("name", "%霞%"));
criteria.add(Restrictions.gt("age", 18));
List<Customer> list = criteria.list();
for (Customer customer : list){
System.out.println(customer);
}
// 4.分頁查詢
Criteria criteria = session.createCriteria(Customer.class);
criteria.setFirstResult(3);
criteria.setMaxResults(3);
List<Customer> list = criteria.list();
for (Customer customer : list){
System.out.println(customer);
}
2.3 SQLQuery查詢接口
1. 基本查詢
SQLQuery sqlQuery = session.createSQLQuery("select * from cst_customer");
List<Object[]> list = sqlQuery.list();
for(Object[] objects : list) {
System.out.println(Arrays.toString(objects));
}
SQLQuery sqlQuery = session.createSQLQuery("select * from cst_customer");
// 封裝到對象中
sqlQuery.addEntity(Customer.class);
List<Customer> list = sqlQuery.list();
for(Customer customer:list){
System.out.println(customer);
}