Spring事务管理

Spring事务管理分为声明式事务管理编程式事务管理,声明式事务管理又分为xml注解两种配置方式。应该优先选择声明式事务,因为声明式事务对程序代码的影响最小,因此最符合非侵入式轻量级容器的理想 。只有在进行少量事务操作时,才应该选择编程式事务管理的方式。

声明式事务管理

xml配置方式

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"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/tx
        https://www.springframework.org/schema/tx/spring-tx.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd
        http://www.springframework.org/schema/context
		https://www.springframework.org/schema/context/spring-context.xsd">
    
    <!-- 导入数据库配置文件 -->
    <context:property-placeholder location="config.properties"/>
    
    <!-- 配置数据源 -->
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
		init-method="init" destroy-method="close">
		<property name="driverClassName"><value>${jdbc_driverClassName}</value></property>
		<property name="url"><value>${jdbc_url}</value></property>
		<property name="username"><value>${jdbc_username}</value></property>
		<property name="password"><value>${jdbc_password}</value></property>
		<!-- ... -->
    </bean>
    
    <!-- 配置jdbcTemplate,使用jdbcTemplate操作数据库 -->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    	<property name="dataSource" ref="dataSource"/>
    </bean>
    
    <bean id="userService" class="com.springdemo.tx.UserService">
    	<property name="jdbcTemplate" ref="jdbcTemplate"/>
    </bean>
    
    <!-- 配置事务管理器 -->
    <bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
    </bean>

    <!-- xml方式的配置声明式事务 start-->
    <tx:advice id="txAdvice" transaction-manager="txManager">
        <tx:attributes>
            <tx:method name="get*" read-only="true"/>
            <tx:method name="*"/>
        </tx:attributes>
    </tx:advice>

    <aop:config>
        <aop:advisor advice-ref="txAdvice" pointcut="execution(* com.springdemo.tx.UserService.*(..))"/>
    </aop:config>
    <!-- xml方式的配置声明式事务 end-->
</beans>

config.properties:

jdbc_driverClassName=com.mysql.jdbc.Driver
jdbc_url=jdbc:mysql://192.168.1.77:3306/testdb?useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true
jdbc_username=root
jdbc_password=root

UserService.java:

public class UserService {
	private JdbcTemplate jdbcTemplate;

	public void addUser(String name, int age){
		jdbcTemplate.update("insert into test007 (name,age) values (name,age)", name, age);
	}
	
	public void updateUserName(String name, int id){
		jdbcTemplate.update("update test007 set name=? where id=?", name, id);
	}
	
	public void updateUserAge(int age, int id){
		jdbcTemplate.update("update test007 set age=? where id=?", age, id);
	}
	
	public void updateUser(String name, int age, int id){
		updateUserName(name, id);
		//模拟异常
		Double.parseDouble("我不是Double,所以会抛出NumberFormatException异常");
		updateUserAge(age, id);
	}

	public JdbcTemplate getJdbcTemplate() {
		return jdbcTemplate;
	}

	public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
		this.jdbcTemplate = jdbcTemplate;
	}
}

main方法:

public static void main(String[] args) {
		ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("spring-config.xml");
		UserService userService = (UserService) ac.getBean("userService");
		userService.updateUser("小王", 18, 1);
}

使用注解方式

Spring配置文件中添加:

	<!-- 开启注解配置 -->
	<tx:annotation-driven transaction-manager="txManager"/>

然后在需要事务管理的方法上添加@Transactional注解

	@Transactional
	public void updateUser(String name, int age, int id){
		updateUserName(name, id);
		//模拟异常
		Double.parseDouble("我不是Double,所以会抛出NumberFormatException异常");
		updateUserAge(age, id);
	}

注意:注解方式和xml方式可以二选一,也可以结合使用,两者没有依赖关系。

异常捕获和回滚

异常捕获会导致事务机制失效,继而不会触发回滚。
可以调用TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()手动回滚。

    @Transactional
	public void updateUser(String name, int age, int id){
		try{
			updateUserName(name, id);
			//模拟异常
			Double.parseDouble("我不是Double,所以会抛出NumberFormatException异常");
			updateUserAge(age, id);
		}catch(NumberFormatException ex){
			//手动回滚
			TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
			ex.printStackTrace();
		}
	}

非RuntimeException时回滚

Spring事务管理机制默认只在抛出RuntimeException时才会触发回滚,可以设置rollbackFor属性来指定其他类型的异常也能回滚。

    //rollbackFor指定抛出Exception类型异常时回滚
    @Transactional(rollbackFor=Exception.class)
	public void updateUser(String name, int age, int id) throws Exception{
		updateUserName(name, id);
		if(id == 1){
			throw new Exception("Id为1的用户不可修改");
		}
		updateUserAge(age, id);
	}

编程式事务管理

修改Spring配置文件:

	<bean id="userService" class="com.springdemo.tx.UserService">
    	<property name="jdbcTemplate" ref="jdbcTemplate"/>
    	<property name="transactionTemplate" ref="transactionTemplate"/>
    </bean>
    
    <bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate"> 
    	<property name="transactionManager" ref="txManager"/>
    </bean>
	
	<!-- 注释声明式事务管理配置 -->
	<!-- <tx:annotation-driven transaction-manager="txManager"/>
    
    <tx:advice id="txAdvice" transaction-manager="txManager">
        <tx:attributes>
            <tx:method name="get*" read-only="true"/>
            <tx:method name="*"/>
        </tx:attributes>
    </tx:advice>

    <aop:config>
        <aop:advisor advice-ref="txAdvice" pointcut="execution(* com.springdemo.tx.UserService.*(..))"/>
    </aop:config> -->

修改UserService:


    private TransactionTemplate transactionTemplate;

    public void updateUser(final String name, final int age, final int id){
		transactionTemplate.execute(new TransactionCallbackWithoutResult() {
		    protected void doInTransactionWithoutResult(TransactionStatus status) {
		    	updateUserName(name, id);
				//模拟异常
				Double.parseDouble("我不是Double,所以会抛出NumberFormatException异常");
				updateUserAge(age, id);
		    }
		});
	}

    public TransactionTemplate getTransactionTemplate() {
		return transactionTemplate;
	}

	public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
		this.transactionTemplate = transactionTemplate;
	}
	
posted @ 2019-07-11 21:15  布禾  阅读(387)  评论(0编辑  收藏  举报