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上