Java EE五大框架之 Hibernate 3

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

1. 之前在講JavaWEB的事務的時候,需要在業務層使用Connection來開啓事務,
* 一種是通過參數的方式傳遞下去
* 另一種是把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);

        }


發佈了49 篇原創文章 · 獲贊 13 · 訪問量 2萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章