Spring+Hibernate集成後事務與Session的一些理解。

Spring的配置文件,在web.xml加載時,就按照類得映射給動態注入了,而且對於其中需要加載的hibernate配置文件,也只加載一次,就把sessionFactory注入給對象中。然後在類中如果需要取得sessionFactory,只需繼承HibernateDaoSupport來取得session.不需要在類中寫set方法,因爲由HibernateDaoSupport中提供。

以往我們單獨使用Hibernate編程式事務,我們都是這樣的步驟:

Session session=HibernateUtils.getSessionFactory().getCurrentSession();

session.beginTranstraction();//開啓事務

session.save(…..);

log.add(log);//假設log.add是在另一個類中的操作數據庫的方法。其中也要用session來操作。HibernateUtils.getSessionFactory().getCurrentSession().save(…..);

session.getTranstraction().commit();

有異常則session.getTranstraction().rollback();

注意:

* 採用getCurrentSession()創建的session會綁定到當前線程中,而採用openSession()

創建的session則不會.所以以上的session是一直都是一個session.

個人認爲:(不管你要執行多少條語句,最後一起commit,就是一個事,也就是一個事務。一個線程只存在一個session)

* 採用getCurrentSession()創建的session在commit或rollback時會自動關閉,而採用openSession()創建的session必須手動關閉

使用getCurrentSession()需要在hibernate.cfg.xml文件中加入如下配置:

*如果使用的是本地事務(jdbc事務)

<propertyname="hibernate.current_session_context_class">thread</property>

* 如果使用的是全局事務(jta事務)

<propertyname="hibernate.current_session_context_class">jta</property>     

採用Spring之後我們採用採用聲明式事務

1、聲明式事務配置

         *配置SessionFactory

         *配置事務管理器

         *事務的傳播特性

         *那些類那些方法使用事務

2、瞭解事務的幾種傳播特性

         1.      PROPAGATION_REQUIRED: 如果存在一個事務,則支持當前事務。如果沒有事務則開啓

         2.      PROPAGATION_SUPPORTS: 如果存在一個事務,支持當前事務。如果沒有事務,則非事務的執行

         3.      PROPAGATION_MANDATORY: 如果已經存在一個事務,支持當前事務。如果沒有一個活動的事務,則拋出異常。

         4.      PROPAGATION_REQUIRES_NEW: 總是開啓一個新的事務。如果一個事務已經存在,則將這個存在的事務掛起。

         5.      PROPAGATION_NOT_SUPPORTED: 總是非事務地執行,並掛起任何存在的事務。

         6.      PROPAGATION_NEVER: 總是非事務地執行,如果存在一個活動事務,則拋出異常

         7.      PROPAGATION_NESTED:如果一個活動的事務存在,則運行在一個嵌套的事務中. 如果沒有活動事務,

              則按TransactionDefinition.PROPAGATION_REQUIRED屬性執行

        舉一個例子:required

public class UserManagerImpl extends HibernateDaoSupport implements UserManager {
    
       private LogManager logManager;
   

       public void addUser(User user) {
            this.getHibernateTemplate().save(user);
            Log log = new Log();
            log.setType("安全日誌");
            log.setDetail("xxx進入系統");
            log.setTime(new Date());
            logManager.addLog(log);
      }

}

public class LogManagerImpl extends HibernateDaoSupport implements LogManager {

    public void addLog(Log log) {
        this.getHibernateTemplate().save(log);
    }
}

在add方法上都配置Required屬性,這樣在執行addUser時,會開啓新事務,然後在方法執行後,再進行事務提交。其中遇到了addlog方法,它發現當前有事務然後就用當前事務,然後噹噹前事務提交後,他也跟着提交。

3、Spring事務的隔離級別

         1.      ISOLATION_DEFAULT: 這是一個PlatfromTransactionManager默認的隔離級別,使用數據庫默認的事務隔離級別.

              另外四個與JDBC的隔離級別相對應

         2.      ISOLATION_READ_UNCOMMITTED: 這是事務最低的隔離級別,它充許令外一個事務可以看到這個事務未提交的數據。

              這種隔離級別會產生髒讀,不可重複讀和幻像讀。

         3.      ISOLATION_READ_COMMITTED: 保證一個事務修改的數據提交後才能被另外一個事務讀取。另外一個事務不能讀取該事務未提交的數據

         4.      ISOLATION_REPEATABLE_READ: 這種事務隔離級別可以防止髒讀,不可重複讀。但是可能出現幻像讀。

              它除了保證一個事務不能讀取另一個事務未提交的數據外,還保證了避免下面的情況產生(不可重複讀)。

         5.      ISOLATION_SERIALIZABLE 這是花費最高代價但是最可靠的事務隔離級別。事務被處理爲順序執行。

              除了防止髒讀,不可重複讀外,還避免了幻像讀。

 

4、編寫業務邏輯方法

         *繼承HibernateDaoSupport類,使用HibernateTemplate來持久化,HibernateTemplate是

          Hibernate Session的輕量級封裝

         *默認情況下運行期異常纔會回滾(包括繼承了RuntimeException子類),普通異常是不會滾的

         *編寫業務邏輯方法時,最好將異常一直向上拋出,在表示層(struts)處理

         *關於事務邊界的設置,通常設置到業務層,不要添加到Dao上 

 

                      


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