zoukankan      html  css  js  c++  java
  • Spring编程式和声明式事务实例讲解

    SnailClimb
    阅读 1160

    Spring编程式和声明式事务实例讲解

    Java面试通关手册(Java学习指南):github.com/Snailclimb/…

    历史回顾: 可能是最漂亮的Spring事务管理详解

    Spring事务管理

    Spring支持两种方式的事务管理:

    • 编程式事务管理: 通过Transaction Template手动管理事务,实际应用中很少使用,
    • 使用XML配置声明式事务: 推荐使用(代码侵入性最小),实际是通过AOP实现

    实现声明式事务的四种方式:

    1. 基于 TransactionInterceptor 的声明式事务: Spring 声明式事务的基础,通常也不建议使用这种方式,但是与前面一样,了解这种方式对理解 Spring 声明式事务有很大作用。

    2. 基于 TransactionProxyFactoryBean 的声明式事务: 第一种方式的改进版本,简化的配置文件的书写,这是 Spring 早期推荐的声明式事务管理方式,但是在 Spring 2.0 中已经不推荐了。

    3. 基于< tx> 和< aop>命名空间的声明式事务管理: 目前推荐的方式,其最大特点是与 Spring AOP 结合紧密,可以充分利用切点表达式的强大支持,使得管理事务更加灵活。

    4. 基于 @Transactional 的全注解方式: 将声明式事务管理简化到了极致。开发人员只需在配置文件中加上一行启用相关后处理 Bean 的配置,然后在需要实施事务管理的方法或者类上使用 @Transactional 指定事务规则即可实现事务管理,而且功能也不必其他方式逊色。

    我们今天要将的是使用编程式以及基于AspectJ的声明式和基于注解的事务方式,实现烂大街的转账业务。

    再来说一下这个案例的思想吧,我们在两次转账之间添加一个错误语句(对应银行断电等意外情况),如果这个时候两次转账不能成功,则说明事务配置正确,否则,事务配置不正确。

    你需要完成的任务:

    • 使用编程式事务管理完成转账业务
    • 使用基于AspectJ的声明式事务管理完成转账业务
    • 使用基于 @Transactional 的全注解方式事务管理完成转账业务

    备注:

    下面的代码是在很久之前,我刚学Sping还没有接触Maven的时候写的,所以我使用的原始添加jar的方式,使用Maven的小伙伴可以自行添加Maven依赖,没有使用Maven的小伙伴直接使用我下面提供的jar包即可。

    jar包地址:链接:pan.baidu.com/s/1tqy-mVKx… 密码:nid0

    项目结构:

    项目结构

    开发工具:

    Myeclipse2017

    SQL:

    
    create table `account` (
    	`username` varchar (99),
    	`salary` int (11)
    ); 
    insert into `account` (`username`, `salary`) values('小王','3000');
    insert into `account` (`username`, `salary`) values('小马','3000');
    

    复制代码

    (1)编程式事务管理

    注意: 通过添加/删除accountMoney() 方法中int i = 10 / 0这个语句便可验证事务管理是否配置正确。

    OrdersDao.java(Dao层)

    package cn.itcast.dao;
    

    import org.springframework.jdbc.core.JdbcTemplate;

    public class OrdersDao {
    // 注入jdbcTemplate模板对象
    private JdbcTemplate jdbcTemplate;

    <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">setJdbcTemplate</span><span class="hljs-params">(JdbcTemplate jdbcTemplate)</span> </span>{
    	<span class="hljs-keyword">this</span>.jdbcTemplate = jdbcTemplate;
    }
    
    <span class="hljs-comment">// 对数据操作的方法不包含业务操作</span>
    <span class="hljs-comment">/**
     * 小王少钱的方法
     */</span>
    <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">reduceMoney</span><span class="hljs-params">()</span> </span>{
    	String sql = <span class="hljs-string">"update account set salary=salary-? where username=?"</span>;
    	jdbcTemplate.update(sql, <span class="hljs-number">1000</span>, <span class="hljs-string">"小王"</span>);
    }
    
    <span class="hljs-comment">/**
     * 小马多钱的方法
     */</span>
    <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">addMoney</span><span class="hljs-params">()</span> </span>{
    	String sql = <span class="hljs-string">"update account set salary=salary+? where username=?"</span>;
    	jdbcTemplate.update(sql, <span class="hljs-number">1000</span>, <span class="hljs-string">"小马"</span>);
    }
    

    }
    复制代码

    OrdersService.java(业务逻辑层)

    package cn.itcast.service;
    
    import org.springframework.transaction.TransactionStatus;
    import org.springframework.transaction.support.TransactionCallback;
    import org.springframework.transaction.support.TransactionTemplate;
    
    import cn.itcast.dao.OrdersDao;
    
    public class OrdersService {
    	// 注入Dao层对象
    	private OrdersDao ordersDao;
    
    	public void setOrdersDao(OrdersDao ordersDao) {
    		this.ordersDao = ordersDao;
    	}
    
    	// 注入TransactionTemplate对象
    	private TransactionTemplate transactionTemplate;
    
    	public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
    		this.transactionTemplate = transactionTemplate;
    	}
    
    	// 调用dao的方法
    	// 业务逻辑,写转账业务
    	public void accountMoney() {
    		transactionTemplate.execute(new TransactionCallback<Object>() {
    
    			@Override
    			public Object doInTransaction(TransactionStatus status) {
    				Object result = null;
    				try {
    					// 小马多1000
    					ordersDao.addMoney();
    					// 加入出现异常如下面int
    					// i=10/0(银行中可能为突然停电等。。。);结果:小马账户多了1000而小王账户没有少钱
    					// 解决办法是出现异常后进行事务回滚
    					int i = 10 / 0;// 事务管理配置后异常已经解决
    					// 小王 少1000
    					ordersDao.reduceMoney();
    				} catch (Exception e) {
    					status.setRollbackOnly();
    					result = false;
    					System.out.println("Transfer Error!");
    				}
    
    				return result;
    			}
    		});
    
    	}
    }
    复制代码

    TestService.java(测试方法)

    package cn.itcast.service;
    
    import org.junit.Test;
    import org.springframework.context.ApplicationContext;
    import org.springframework.context.support.ClassPathXmlApplicationContext;
    
    public class TestService {
    	@Test
    	public void testAdd() {
    		ApplicationContext context = new ClassPathXmlApplicationContext(
    				"beans.xml");
    		OrdersService userService = (OrdersService) context
    				.getBean("ordersService");
    		userService.accountMoney();
    	}
    }
    
    复制代码

    配置文件:

    <?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:context="http://www.springframework.org/schema/context"
    	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.5.xsd  
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd  
    http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd  
    http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
    	<!-- 配置c3po连接池 -->
    	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
    		<!-- 注入属性值 -->
    		<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
    		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
    		<property name="user" value="root"></property>
    		<property name="password" value="153963"></property>
    	</bean>
    	<!-- 编程式事务管理 -->
    	<!-- 配置事务管理器 -->
    	<bean id="dataSourceTransactionManager"
    		class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    		<!-- 注入dataSource -->
    		<property name="dataSource" ref="dataSource"></property>
    	</bean>
    
    	<!-- 配置事务管理器模板 -->
    	<bean id="transactionTemplate"
    		class="org.springframework.transaction.support.TransactionTemplate">
    		<!-- 注入真正进行事务管理的事务管理器,name必须为 transactionManager否则无法注入 -->
    		<property name="transactionManager" ref="dataSourceTransactionManager"></property>
    	</bean>
    
    	<!-- 对象生成及属性注入 -->
    	<bean id="ordersService" class="cn.itcast.service.OrdersService">
    		<property name="ordersDao" ref="ordersDao"></property>
    		<!-- 注入事务管理的模板 -->
    		<property name="transactionTemplate" ref="transactionTemplate"></property>
    	</bean>
    
    	<bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
    		<property name="jdbcTemplate" ref="jdbcTemplate"></property>
    	</bean>
    	<!-- JDBC模板对象 -->
    	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    		<property name="dataSource" ref="dataSource"></property>
    	</bean>
    </beans> 
    复制代码

    (2)基于AspectJ的声明式事务管理

    OrdersService.java(业务逻辑层)

    package cn.itcast.service;
    

    import cn.itcast.dao.OrdersDao;

    public class OrdersService {
    private OrdersDao ordersDao;

    <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">setOrdersDao</span><span class="hljs-params">(OrdersDao ordersDao)</span> </span>{
    	<span class="hljs-keyword">this</span>.ordersDao = ordersDao;
    }
    
    <span class="hljs-comment">// 调用dao的方法</span>
    <span class="hljs-comment">// 业务逻辑,写转账业务</span>
    <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">accountMoney</span><span class="hljs-params">()</span> </span>{
    	<span class="hljs-comment">// 小马多1000</span>
    	ordersDao.addMoney();
    	<span class="hljs-comment">// 加入出现异常如下面int i=10/0(银行中可能为突然停电等。。。);结果:小马账户多了1000而小王账户没有少钱</span>
    	<span class="hljs-comment">// 解决办法是出现异常后进行事务回滚</span>
    	<span class="hljs-keyword">int</span> i = <span class="hljs-number">10</span> / <span class="hljs-number">0</span>;<span class="hljs-comment">// 事务管理配置后异常已经解决</span>
    	<span class="hljs-comment">// 小王 少1000</span>
    	ordersDao.reduceMoney();
    }
    

    }

    复制代码

    配置文件:

    	<!-- 配置c3po连接池 -->
    	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
    		<!-- 注入属性值 -->
    		<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
    		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
    		<property name="user" value="root"></property>
    		<property name="password" value="153963"></property>
    	</bean>
    	<!-- 第一步:配置事务管理器 -->
    	<bean id="dataSourceTransactionManager"
    		class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    		<!-- 注入dataSource -->
    		<property name="dataSource" ref="dataSource"></property>
    	</bean>
    
    	<!-- 第二步:配置事务增强 -->
    	<tx:advice id="txadvice" transaction-manager="dataSourceTransactionManager">
    		<!-- 做事务操作 -->
    		<tx:attributes>
    			<!-- 设置进行事务操作的方法匹配规则 -->
    			<!-- account开头的所有方法 -->
    	        <!--
    	          propagation:事务传播行为; 
    	          isolation:事务隔离级别;
    	          read-only:是否只读;
    	          rollback-for:发生那些异常时回滚 
    	          timeout:事务过期时间
    	         -->
    			<tx:method name="account*" propagation="REQUIRED"
    				isolation="DEFAULT" read-only="false" rollback-for="" timeout="-1" />
    		</tx:attributes>
    	</tx:advice>
    
    	<!-- 第三步:配置切面 切面即把增强用在方法的过程 -->
    	<aop:config>
    		<!-- 切入点 -->
    		<aop:pointcut expression="execution(* cn.itcast.service.OrdersService.*(..))"
    			id="pointcut1" />
    		<!-- 切面 -->
    		<aop:advisor advice-ref="txadvice" pointcut-ref="pointcut1" />
    	</aop:config>
    
    
    	<!-- 对象生成及属性注入 -->
    	<bean id="ordersService" class="cn.itcast.service.OrdersService">
    		<property name="ordersDao" ref="ordersDao"></property>
    	</bean>
    	<bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
    		<property name="jdbcTemplate" ref="jdbcTemplate"></property>
    	</bean>
    	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    		<property name="dataSource" ref="dataSource"></property>
    	</bean>
    复制代码

    (3)基于注解的方式

    OrdersService.java(业务逻辑层)

    package cn.itcast.service;
    

    import org.springframework.transaction.annotation.Isolation;
    import org.springframework.transaction.annotation.Propagation;
    import org.springframework.transaction.annotation.Transactional;

    import cn.itcast.dao.OrdersDao;

    @Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false, timeout = -1)
    public class OrdersService {
    private OrdersDao ordersDao;

    <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">setOrdersDao</span><span class="hljs-params">(OrdersDao ordersDao)</span> </span>{
    	<span class="hljs-keyword">this</span>.ordersDao = ordersDao;
    }
    
    <span class="hljs-comment">// 调用dao的方法</span>
    <span class="hljs-comment">// 业务逻辑,写转账业务</span>
    <span class="hljs-function"><span class="hljs-keyword">public</span> <span class="hljs-keyword">void</span> <span class="hljs-title">accountMoney</span><span class="hljs-params">()</span> </span>{
    	<span class="hljs-comment">// 小马多1000</span>
    	ordersDao.addMoney();
    	<span class="hljs-comment">// 加入出现异常如下面int i=10/0(银行中可能为突然停电等。。。);结果:小马账户多了1000而小王账户没有少钱</span>
    	<span class="hljs-comment">// 解决办法是出现异常后进行事务回滚</span>
    	<span class="hljs-comment">// int i = 10 / 0;// 事务管理配置后异常已经解决</span>
    	<span class="hljs-comment">// 小王 少1000</span>
    	ordersDao.reduceMoney();
    }
    

    }
    复制代码

    配置文件:

    	<!-- 配置c3po连接池 -->
    	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
    		<!-- 注入属性值 -->
    		<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
    		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
    		<property name="user" value="root"></property>
    		<property name="password" value="153963"></property>
    	</bean>
    	<!-- 第一步:配置事务管理器 (和配置文件方式一样)-->
    	<bean id="dataSourceTransactionManager"
    		class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    		<!-- 注入dataSource -->
    		<property name="dataSource" ref="dataSource"></property>
    	</bean>
    	<!-- 第二步: 开启事务注解 -->
    	<tx:annotation-driven transaction-manager="dataSourceTransactionManager" />
    	<!-- 第三步 在方法所在类上加注解 -->
    	
    	
    	<!-- 对象生成及属性注入 -->
    	<bean id="ordersService" class="cn.itcast.service.OrdersService">
    		<property name="ordersDao" ref="ordersDao"></property>
    	</bean>
    	<bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
    		<property name="jdbcTemplate" ref="jdbcTemplate"></property>
    	</bean>
    	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    		<property name="dataSource" ref="dataSource"></property>
    	</bean>
    复制代码

    欢迎关注我的微信公众号: "Java面试通关手册" (一个有温度的微信公众号,期待与你共同进步~~~坚持原创,分享美文,分享各种Java学习资源):

    关注下面的标签,发现更多相似文章
    评论

  • 相关阅读:
    .net 关于路径的总结
    asp.net Base64加解密
    asp.net中的<% %>,<%= %>,<%# %><%$ %>的使用
    asp.net Swiper 轮播动画
    ASP.NET中Literal控件的使用方法(用于向网页中动态添加内容)
    asp.net 获取表单中控件的值
    rgb值转换成16进制
    关于background全解
    移动端的性能陷阱
    原生JS实现雪花特效
  • 原文地址:https://www.cnblogs.com/jobs-lgy/p/9915456.html
Copyright © 2011-2022 走看看