Spring学习笔记之九--声明式事务

一、事务的传播特性:
  前六个是标准的,在.net中也有,后面的是spring专有的。最重要的是第一种。用的最多
 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 属性执行

 

表格示例(事务t1 ,t2)


 

二、数据库隔离级别介绍
 

隔离级别

是否存在脏读

是否存在不可重复读

是否存在幻读

未提交读(没有提交别人就能读到)

Read uncommited

Yes(没有提交别人就能读到)

Yes(假设我原来数据查上来是张三,当我一刷新变成李四了)

可用悲观锁来锁住,不让别人修改!

Yes(假设我在查18-20岁的学生人数5条,当我一刷新,突然多了3条)

提交读(大部分数据库的默认设置如Oracle

Read commited

NO

Yes

Yes

可重复读(mysql的默认级别)

Repeatable Read

NO

NO

Yes

序列化读

Serialiazble

NO

NO

No

 

三、Spring事务的隔离级别
 1. ISOLATION_DEFAULT: 这是一个PlatfromTransactionManager默认的隔离级别,使用数据库默认的事务隔离级别.
      另外四个与JDBC的隔离级别相对应
 2. ISOLATION_READ_UNCOMMITTED: 这是事务最低的隔离级别,它充许令外一个事务可以看到这个事务未提交的数据。
      这种隔离级别会产生脏读,不可重复读和幻像读。
 3. ISOLATION_READ_COMMITTED: 保证一个事务修改的数据提交后才能被另外一个事务读取。另外一个事务不能读取该事务未提交的数据
 4. ISOLATION_REPEATABLE_READ: 这种事务隔离级别可以防止脏读,不可重复读。但是可能出现幻像读。
      它除了保证一个事务不能读取另一个事务未提交的数据外,还保证了避免下面的情况产生(不可重复读)。
 5. ISOLATION_SERIALIZABLE 这是花费最高代价但是最可靠的事务隔离级别。事务被处理为顺序执行。
      除了防止脏读,不可重复读外,还避免了幻像读。

 

、声明式事务配置步骤
 * 配置SessionFactory
 * 配置事务管理器
 * 事务的传播特性
 * 那些类那些方法使用事务

 

、配置文件
1、准备工作,在配置文件中若打上<tx:,若没有提示信息帮助,处理如下(Eclipse/Windows/perfermance/MyEclipse/files and Editor/xml/xml catalog/user specified

Engtries/add/跳出一个有三个文本域的对话框,中间的先选中url,然后对最上面的文本框采用浏览文件系统的方式,选中spring-framework-2.0/dist/resources/spring-tx-

2.0.xsd)
此时这三个文本域都有内容了。
修改文本域内容:中间的改选为“location”,最下面的在原来的基础上,在末尾加上\spring-tx-2.0.xsd
即加上你上面选定(选中spring-framework-2.0/dist/resources/spring-tx-2.0.xsd)的schemel的名字。
然后关闭再进入就ok了。


2、事务配置文件applicationContext-common.xml

 

下面内部的 <!-- 配置sessionFactory -->中
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">

LocalSessionFactoryBean是spring已经实现好的类,配置一下就可以。注入大工厂中。

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xmlns:aop="http://www.springframework.org/schema/aop"
      xmlns:tx="http://www.springframework.org/schema/tx"
      xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">
 <!-- 配置sessionFactory -->
 <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
  <property name="configLocation">
   <value>classpath:hibernate.cfg.xml</value>
  </property> 
 </bean>           
 
 <!-- 配置事务管理器 -->
 <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
  <property name="sessionFactory">
   <ref bean="sessionFactory"/>
  </property> 
 </bean>
 
 <!-- 配置事务的传播特性 -->
 <tx:advice id="txAdvice" transaction-manager="transactionManager">
  <tx:attributes>
   <tx:method name="add*" propagation="REQUIRED"/>
   <tx:method name="del*" propagation="REQUIRED"/>
   <tx:method name="modify*" propagation="REQUIRED"/>
   <tx:method name="*" read-only="true"/>
  </tx:attributes>
 </tx:advice>
 
 <!-- 那些类的哪些方法参与事务 -->
 <aop:config>
  <aop:pointcut id="allManagerMethod" expression="execution(* com.bjsxt.usermgr.manager.*.*(..))"/>
  <aop:advisor pointcut-ref="allManagerMethod" advice-ref="txAdvice"/>
 </aop:config>
</beans>



 

3、配置文件applicationContext-bean.xml

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"
	     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	     xmlns:aop="http://www.springframework.org/schema/aop"
	     xmlns:tx="http://www.springframework.org/schema/tx"
	     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">
	
	<bean id="userManager" class="com.bjsxt.usermgr.manager.UserManagerImpl">
		<property name="sessionFactory" ref="sessionFactory"/>
		<property name="logManager" ref="logManager"/>
	</bean>
	
	<bean id="logManager" class="com.bjsxt.usermgr.manager.LogManagerImpl">
		<property name="sessionFactory" ref="sessionFactory"/>
	</bean>
</beans>

 

 

 

六、编写业务逻辑方法
 * 继承HibernateDaoSupport类,使用HibernateTemplate来持久化,HibernateTemplate是
   Hibernate Session的轻量级封装
 * 默认情况下运行期异常才会回滚(包括继承了RuntimeException子类,甚至可以指定抛出哪个异常回滚),普通异常是不回滚的。
 * 编写业务逻辑方法时,最好将异常一直向上抛出,在表示层(struts)处理
 * 关于事务边界的设置,通常设置到业务层,不要添加到Dao上 
 七、部分代码(整体见附件)

public class Client {

	public static void main(String[] args) {
		User user = new User();
		user.setName("张三");
		
		BeanFactory factory = new ClassPathXmlApplicationContext("applicationContext-*.xml");
		UserManager userManager = (UserManager)factory.getBean("userManager");
		try {
			userManager.addUser(user);
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

 

 

注:例子中繼承HibernateDaoSupport 的好處是,因为以前使用hibernate还需要对session进行管理,在此处可以讓spring完成自動的對session的管理。若不繼承也可以的,我們可以像以前一樣開啟session,關閉session的如session = HibernateUtils.getSessionFactory().getCurrentSession();                           session.beginTransaction();。。。。。

public class LogManagerImpl extends HibernateDaoSupport implements LogManager {

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

 

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);
//		throw new java.lang.RuntimeException();
//	}

	public void addUser(User user)
	throws Exception {
		this.getHibernateTemplate().save(user);
		Log log = new Log();
		log.setType("安全日志");
		log.setDetail("xxx进入系统");
		log.setTime(new Date());
		logManager.addLog(log);
		throw new Exception();
	}
	
	public void setLogManager(LogManager logManager) {
		this.logManager = logManager;
	}
}

 

 

public class HibernateUtils {

	private static SessionFactory factory;
	
	static {
		try {
			Configuration cfg = new Configuration().configure();
			factory = cfg.buildSessionFactory();
		}catch(Exception e) {
			e.printStackTrace();
		}
	}
	
	public static SessionFactory getSessionFactory() {
		return factory;
	}
	
	public static Session getSession() {
		return factory.openSession();
	}
	
	public static void closeSession(Session session) {
		if (session != null) {
			if (session.isOpen()) {
				session.close();
			}
		}
	}
}

 

你可能感兴趣的:(spring,AOP,bean,Hibernate,配置管理)