Spring之事务

Spring提供了如下两种事务管理方式。

1)        编程式事务管理:即使用Spring的编程式事务,程序也可直接获取容器中的transactionManagerBean,该Bean总是PlatformTransactionManager的实例,所以可以通过该接口提供的三个方法来开始事务、提交事务和回滚事务。

2)        声明式事务管理:无须在Java程序中书写任何事务操作代码,而是通过XML文件文件中为业务组件配置事务代理(AOP代理的一种),AOP为事务代理所织入的增强也由Spring提供——在目标方法执行之前,织入开台事务;在目标方法执行之后,织入结束事务。

1. 使用XML Schema配置事务策略

       Spring提供了简洁的事务配置策略,即使用<tx:advice…/>。配置<tx:advice…/>元素时除需要transaction-manager属性指定事务管理器之外,还需要配置一个<attributes…/>子元素,该子元素里又可包含多个<method…/>子元素。其实,配置<tx:advice…/>元素的重点就是配置<method…/>子元素,实际上每个<method…/>子元素都为一批方法指定了所需要的事务定义,包含事务传播属性、事务隔离属性、事务超时属性、只读事务、对指定异常回滚、对指定异常不回滚等。

配置<method…/>子元素可以指定如下属性。

1)        name:必选属性,与事务语义关联的方法名。

2)        propagation:指定事务传播行为,该属性值可为Propagation枚举的任一枚举值。该属性的默认值为:Propagation.REQUIRED。

3)        isolation:指定事隔离级别,该属性值可为Isolation枚举的任一枚举值。该属性的默认值为:Isolation.DEFAULT。

4)        timeout:指定事务超时的时间(以秒为单位),指定-1意味着不超时,该属性的默认值是-1.

5)        read-only:指定事务是否只读。该属性默认值是false。

6)        rollback-for:指定触发事务回滚的异常类(应使用全限定类名),该属性可指定多个异常类,多个异常类之间用英文逗号隔开。

7)        no-rollback-for:指定不触发事务回滚的异常类(应使用全限定类名),该属性可指定多个异常类,多个异常类之间以英文逗号隔开。

<method…/>子元素的propagation属性用于指定事务传播行为,Spring支持的事务传播行为如下:

1)        PROPAGTION_MANDATORY:要求调用该方法的线程必须处于事务环境中,否则抛出异常。

2)        PROPAGATION_NEWSTED:即使执行该方法的线程已处于事务环境中,也依然启动新的事务,方法在嵌套的事务里执行;即使执行该方法的线程并未处于事务环境中,也启动新的事务,然后执行该方法,此时与PROPAGATION_REQUIRED相同。

3)        PROPAGATION_NEVER:不允许调用该方法的线程处于事务环境中,如果调用该方法的线程处于事务环境中,则抛出异常。

4)        PROPAGATION_NOT_SUPPORTED:如果调用该方法的线程处于事务环境中,则先暂停当前事务,然后执行该方法。

5)        PROPAGATION_REQUIRED:要求在事务环境中执行该方法,如果当前执行线程已处于事务环境中,则直接调用;如果当前执行线程不处于事务环境中,则启动新的事务后执行该方法。

6)        PROPAGATION_REQUIRES-NEW:该方法要求在新的事务环境中执行,如果当前执行线程已处于事务环境中,则先暂停当前事务,启动新事务后执行该方法;如果当前调用线程不处于事务环境中,则启动新的事务后执行该方法。

7)        PROPAGATION_SUPPORTS:如果当前执行线程处于事务环境中,则使用当前事务,否则不使用事务。

下面我们定义个NewDaoImpl组件,这个组件包含一个insert()方法,该方法同时插入两条记录,但插入的第二条记录将会违反唯一键约束,从而引发异常。

public class NewsDaoImpl implements NewsDao
{
	private DataSource ds;
	public void setDs(DataSource ds)
	{
		this.ds = ds;
	}
	public void insert(String title, String content)
	{
		JdbcTemplate jt = new JdbcTemplate(ds);
		jt.update("insert into news_inf"
			+ " values(null , ? , ?)"
			, title , content);
		// 两次插入的数据违反唯一键约束
		jt.update("insert into news_inf"
			+ " values(null , ? , ?)"
			, title , content);
		// 如果没有事务控制,则第一条记录可以被插入
		// 如果增加事务控制,将发现第一条记录也插不进去。
	}
}

下面是本应用示例所使用的配置文件。

<?xml version="1.0" encoding="GBK"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns="http://www.springframework.org/schema/beans"
	xmlns:p="http://www.springframework.org/schema/p"
	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-4.0.xsd
	http://www.springframework.org/schema/aop
	http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
	http://www.springframework.org/schema/tx
	http://www.springframework.org/schema/tx/spring-tx-4.0.xsd">
	<!-- 定义数据源Bean,使用C3P0数据源实现,并注入数据源的必要信息 -->
	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
		destroy-method="close"
		p:driverClass="com.mysql.jdbc.Driver"
		p:jdbcUrl="jdbc:mysql://localhost/spring"
		p:user="root"
		p:password="32147"
		p:maxPoolSize="40"
		p:minPoolSize="2"
		p:initialPoolSize="2"
		p:maxIdleTime="30"/>
	<!-- 配置JDBC数据源的局部事务管理器,使用DataSourceTransactionManager 类 -->
	<!-- 该类实现PlatformTransactionManager接口,是针对采用数据源连接的特定实现-->
	<!-- 配置DataSourceTransactionManager时需要依注入DataSource的引用 -->
	<bean id="transactionManager" 
		class="org.springframework.jdbc.datasource.DataSourceTransactionManager"
		p:dataSource-ref="dataSource"/>
	<!-- 配置一个业务逻辑Bean -->
	<bean id="newsDao" class="com.owen.app.dao.impl.NewsDaoImpl"
		p:ds-ref="dataSource"/>
	<!-- 配置事务增强处理Bean,指定事务管理器 -->
	<tx:advice id="txAdvice" 
		transaction-manager="transactionManager">
		<!-- 用于配置详细的事务语义 -->
		<tx:attributes>
			<!-- 所有以'get'开头的方法是read-only的 -->
			<tx:method name="get*" read-only="true"/>
			<!-- 其他方法使用默认的事务设置,指定超时时长为5秒 -->
			<tx:method name="*" isolation="DEFAULT"
				propagation="REQUIRED" timeout="5"/>
		</tx:attributes>
	</tx:advice>
	<!-- AOP配置的元素 -->
	<aop:config>
		<!-- 配置一个切入点,匹配com.owen.app.dao.impl包下
			所有以Impl结尾的类里、所有方法的执行 -->
		<aop:pointcut id="myPointcut"
			expression="execution(* com.owen.app.dao.impl.*Impl.*(..))"/>
		<!-- 指定在myPointcut切入点应用txAdvice事务增强处理 -->
		<aop:advisor advice-ref="txAdvice" 
			pointcut-ref="myPointcut"/>
	</aop:config>
</beans>

上面的例子中,NewDaoImpl其实就是继承NewDao的Bean,并调用它的insert()方法,可以看到该方法已经具有了事务性。

 public class SpringTest
{
	public static void main(String[] args)
	{
		// 创建Spring容器
		ApplicationContext ctx = new
			ClassPathXmlApplicationContext("beans.xml");
		// 获取事务代理Bean
		NewsDao dao = (NewsDao)ctx
			.getBean("newsDao" , NewsDao.class);
		// 执行插入操作
		dao.insert("Java" , "William");
	}
}

2. 使用@Transactional

如果使用 @Transational修饰Bean类,则表明这些事务设置对整个Bean类起作用;如果使用@Transactional修饰Bean类的某个方法,则表明这些事务设置只对该方法有效。使用@Transactioln修饰时可用指定如下:

1)        isolation:用于指定事务的隔离级别。默认为底层事务的隔离级别。

2)        noRollbackFor:指定遇到特定异常时强制不回滚事务。

3)        noRollbackForClassName:指定遇到特定的多个异常时强制不回滚事务。该属性值可以指定多个异常类名。

4)        propagation:指定事务传播行为。

5)        readOnly:指定事务是否只读。

6)        rallbackForClassName:指定遇到特定的多个异常时强制回滚事务。该属性值可以值定多个异常类名。

7)        timeout:指定事务的超时进长。

下面使用@Transactional修饰需要添加事务的方法。

public class NewsDaoImpl implements NewsDao
{
	private DataSource ds;
	public void setDs(DataSource ds)
	{
		this.ds = ds;
	}
	@Transactional(propagation=Propagation.REQUIRED ,
		isolation=Isolation.DEFAULT , timeout=5)
	public void insert(String title, String content)
	{
		JdbcTemplate jt = new JdbcTemplate(ds);
		jt.update("insert into news_inf"
			+ " values(null , ? , ?)"
			, title , content);
		// 两次插入的数据违反唯一键约束
		jt.update("insert into news_inf"
			+ " values(null , ? , ?)"
			, title , content);
		// 如果没有事务控制,则第一条记录可以被插入
		// 如果增加事务控制,将发现第一条记录也插不进去。
	}
}

上面Bean类中的insert()方法使用了@Transaction修饰,表明该方法具有事务性。仅使用这个注解修饰还不够,还需要让Spring根据注解来配置事务代理,所以还需要在Spring配置文件中增加下面配置。

<?xml version="1.0" encoding="GBK"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns="http://www.springframework.org/schema/beans"
	xmlns:p="http://www.springframework.org/schema/p"
	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-4.0.xsd
	http://www.springframework.org/schema/aop
	http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
	http://www.springframework.org/schema/tx
	http://www.springframework.org/schema/tx/spring-tx-4.0.xsd">
	<!-- 定义数据源Bean,使用C3P0数据源实现,并注入数据源的必要信息 -->
	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
		destroy-method="close"
		p:driverClass="com.mysql.jdbc.Driver"
		p:jdbcUrl="jdbc:mysql://localhost/spring"
		p:user="root"
		p:password="32147"
		p:maxPoolSize="40"
		p:minPoolSize="2"
		p:initialPoolSize="2"
		p:maxIdleTime="30"/>

	<!-- 配置一个业务逻辑Bean -->
	<bean id="newsDao" class="com.owen.app.dao.impl.NewsDaoImpl"
		p:ds-ref="dataSource"/>

	<!-- 配置JDBC数据源的局部事务管理器,使用DataSourceTransactionManager 类 -->
	<!-- 该类实现PlatformTransactionManager接口,是针对采用数据源连接的特定实现-->
	<!-- 配置DataSourceTransactionManager时需要依注入DataSource的引用 -->
	<bean id="transactionManager" 
		class="org.springframework.jdbc.datasource.DataSourceTransactionManager"
		p:dataSource-ref="dataSource"/>
	<!-- 根据Annotation来生成事务代理 -->
	<tx:annotation-driven transaction-manager="transactionManager"/>

</beans>






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值