zoukankan      html  css  js  c++  java
  • SpringMVC+MyBatis 事务管理一

    前言

            spring事务管理包含两种情况,编程式事务、声明式事务。而声明式事务又包括基于注解@Transactional和tx+aop的方式。那么本文先分析编程式注解事务和基于注解的声明式事务。 编程式事务管理使用TransactionTemplate或者PlatformTransactionManager。对于编程式事务spring推荐使用TransactionTemplate。

    一、编程式事务

         spring事务特性

         spring中所有的事务策略类都继承自org.springframework.transaction.PlatformTransactionManager接口

    public interface PlatformTransactionManager {
    
        TransactionStatus getTransaction(TransactionDefinition definition) throws TransactionException;
    
        void commit(TransactionStatus status) throws TransactionException;
    
        
        void rollback(TransactionStatus status) throws TransactionException;
    
    }

          编程式事务TransactionTemplate需要手动在代码中处理事务,一般不推荐使用,也不符合spring的思想,因为它直接耦合代码,但各有利弊。先看下TransactionTemplate的源码。

    public class TransactionTemplate extends DefaultTransactionDefinition
    		implements TransactionOperations, InitializingBean {
    
    	
    	protected final Log logger = LogFactory.getLog(getClass());
    
    	private PlatformTransactionManager transactionManager;
    
    	public TransactionTemplate() {
    	}
    
    	public TransactionTemplate(PlatformTransactionManager transactionManager) {
    		this.transactionManager = transactionManager;
    	}
    
    	
    	public TransactionTemplate(PlatformTransactionManager transactionManager, TransactionDefinition transactionDefinition) {
    		super(transactionDefinition);
    		this.transactionManager = transactionManager;
    	}
    
    	public void setTransactionManager(PlatformTransactionManager transactionManager) {
    		this.transactionManager = transactionManager;
    	}
    
    	public PlatformTransactionManager getTransactionManager() {
    		return this.transactionManager;
    	}
    
    	@Override
    	public void afterPropertiesSet() {
    		if (this.transactionManager == null) {
    			throw new IllegalArgumentException("Property 'transactionManager' is required");
    		}
    	}
    
    
    	@Override
    	public <T> T execute(TransactionCallback<T> action) throws TransactionException {
    		if (this.transactionManager instanceof CallbackPreferringPlatformTransactionManager) {
    			return ((CallbackPreferringPlatformTransactionManager) this.transactionManager).execute(this, action);
    		}
    		else {
    			TransactionStatus status = this.transactionManager.getTransaction(this);
    			T result;
    			try {
    				result = action.doInTransaction(status);
    			}
    			catch (RuntimeException ex) {
    				// Transactional code threw application exception -> rollback
    				rollbackOnException(status, ex);
    				throw ex;
    			}
    			catch (Error err) {
    				// Transactional code threw error -> rollback
    				rollbackOnException(status, err);
    				throw err;
    			}
    			catch (Throwable ex) {
    				// Transactional code threw unexpected exception -> rollback
    				rollbackOnException(status, ex);
    				throw new UndeclaredThrowableException(ex, "TransactionCallback threw undeclared checked exception");
    			}
    			this.transactionManager.commit(status);
    			return result;
    		}
    	}
    
    	
    	private void rollbackOnException(TransactionStatus status, Throwable ex) throws TransactionException {
    		logger.debug("Initiating transaction rollback on application exception", ex);
    		try {
    			this.transactionManager.rollback(status);
    		}
    		catch (TransactionSystemException ex2) {
    			logger.error("Application exception overridden by rollback exception", ex);
    			ex2.initApplicationException(ex);
    			throw ex2;
    		}
    		catch (RuntimeException ex2) {
    			logger.error("Application exception overridden by rollback exception", ex);
    			throw ex2;
    		}
    		catch (Error err) {
    			logger.error("Application exception overridden by rollback error", ex);
    			throw err;
    		}
    	}
    
    }
    

      从上面的代码可以看到核心方法是execute,该方法入参TransactionCallback<T>。查看TransactionCallback源码:

    public interface TransactionCallback<T> { 
        T doInTransaction(TransactionStatus status); 
    }
    

     那么以上两个源码可以确定我们使用编程式事务管理的方式也就是自己需要重写doInTransaction()。OK,那么我们手动使用TransactionTemplate处理下。

    1、先配置transactionmanager

     <!--事务管理器 -->
        <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
            <property name="dataSource" ref="dataSource"/>
        </bean>

    2、配置transactionTemplate

     <!--编程式事务,推荐使用TransactionTemplate-->
        <bean id="transactionTemplate"
              class="org.springframework.transaction.support.TransactionTemplate">
            <property name="transactionManager" ref="transactionManager"/>
        </bean>

    3、业务代码处理

     @Autowired
        private TransactionTemplate transactionTemplate;
    
        public int insertUser2(final User user) {
            Integer i= (Integer) this.transactionTemplate.execute(new TransactionCallback() {
                public Object doInTransaction(TransactionStatus transactionStatus) {
    
                    int i = userMapper.insertUser(user);
                    if (i > 0) {
                        System.out.println("success");
                    }
                    int j = 10 / 0;
    
                    return i;
    
                }
            });
    
            return i;
        }

      业务代码中我们使用by zero的异常故意抛出,你会发现能继续打印success,当你断点debug时,你会发现数据库一直是锁定状态,直到你程序执行完。如下图:

    二、基于Transactional注解的事务管理

        当前应该是使用最清爽的事务管理方式了,也符合spring的理念,非入侵代码的方式。

    1、配置

     <!--事务管理器 -->
        <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
            <property name="dataSource" ref="dataSource"/>
        </bean>
    
     <!-- 使用注解事务,需要添加Transactional注解属性 -->
        <tx:annotation-driven transaction-manager="transactionManager"/>
    
     <!--启用最新的注解器、映射器-->
        <mvc:annotation-driven/>

    2、配置后只需要在要处理的地方加上Transactional注解,而且Transactional注解的方式可以应用在类上,也可以应用在方法上,当然只针对public方法。

    3、业务代码处理

      @Transactional
        public int insertUser(User user) {
            int i = userMapper.insertUser(user);
            if (i > 0) {
                System.out.println("success");
            }
            int j = 10 / 0;
    
            return i;
        }

      参考资料

    http://www.cnblogs.com/wyisprogramming/p/6944878.html

    http://www.cnblogs.com/xusir/p/3650522.html

  • 相关阅读:
    买了一个 站立式办公 桌子。
    AutoCppHeader AutoHeader 自动根据CPP 或C文件 来生成头文件。
    cplusplus 库 在线管理; 类似于 python的 pip install 、nodejs 的npm模块
    var this.value 可在任意地方 声明 类成员变量。如果可以利用在C++ 那该有多好啊
    最近遇到了 timer1sec 定时调用的函数,出现了 时间久了,就不是每秒一次了,可能会慢的情况。如何解决呢?
    【我的愿景】每个程序员都应该学习编译原理,而每个人都应该学习 编程 和 设计。
    [已解决]下载chromium源码 download_from_google_storage 无法下载文件
    突发奇想-取缔宏定义,用另外一种语言来做C++的部分外层封装
    我讨厌Apple Safari浏览器的一些地方。不想用
    unix时间戳time_t与UTC时区的关系
  • 原文地址:https://www.cnblogs.com/sword-successful/p/7274929.html
Copyright © 2011-2022 走看看