Spring_11 Transaction(事務) 使用(XML 配置方式)

weixin_33890499發表於2017-10-07

在Spring 中使用Transaction (事務),在很多時候當一個動作需要修改多個相互有聯絡表中的資料,如果其中一條資料修改失敗,要對已經修改的資料進行回滾操作,這時就要使用Transaction 進行回滾。

Spring 內建事務管理器

Spring提供了許多內建事務管理器實現:

Class 描述
DataSourceTransactionManager 位於org.springframework.jdbc.datasource包中,資料來源事務管理器,提供對單個javax.sql.DataSource事務管理,用於Spring JDBC抽象框架、iBATIS或MyBatis框架的事務管理
JdoTransactionManager 位於org.springframework.orm.jdo包中,提供對單個javax.jdo.PersistenceManagerFactory事務管理,用於整合JDO框架時的事務管理
JpaTransactionManager 位於org.springframework.orm.jpa包中,提供對單個javax.persistence.EntityManagerFactory事務支援,用於整合JPA實現框架時的事務管理
HibernateTransactionManager 位於org.springframework.orm.hibernate3包中,提供對單個org.hibernate.SessionFactory事務支援,用於整合Hibernate框架時的事務管理;該事務管理器只支援Hibernate3+版本,且Spring3.0+版本只支援Hibernate 3.2+版本
JtaTransactionManager 位於org.springframework.transaction.jta包中,提供對分散式事務管理的支援,並將事務管理委託給Java EE應用伺服器事務管理器
OC4JjtaTransactionManager 位於org.springframework.transaction.jta包中,Spring提供的對OC4J10.1.3+應用伺服器事務管理器的介面卡,此介面卡用於對應用伺服器提供的高階事務的支援
WebSphereUowTransactionManager 位於org.springframework.transaction.jta包中,Spring提供的對WebSphere 6.0+應用伺服器事務管理器的介面卡,此介面卡用於對應用伺服器提供的高階事務的支援
WebLogicJtaTransactionManager 位於org.springframework.transaction.jta包中,Spring提供的對WebLogic 8.1+應用伺服器事務管理器的介面卡,此介面卡用於對應用伺服器提供的高階事務的支援

在Spring 中使用Transaction

引入jar

在專案中要引入 spring-tx-4.2.4.RELEASE.jar

在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: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.xsd
        http://www.springframework.org/schema/context 
        http://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/tx 
        http://www.springframework.org/schema/tx/spring-tx.xsd
        http://www.springframework.org/schema/aop 
        http://www.springframework.org/schema/aop/spring-aop.xsd">

</beans>

這裡基本整理了前面提到的所有約束配置。

在XML 中配置事務

以下內容都是基於JDBC

  1. 配置事務管理器
<bean id="cPDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
    <property name="driverClass" value="com.mysql.jdbc.Driver" />
    <property name="jdbcUrl" value="jdbc:mysql://127.0.0.1:3306/spring_db?useUnicode=true&characterEncoding=UTF-8" />
    <property name="user" value="root" />
    <property name="password" value="root" />
</bean>
<!-- 配置事務管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <!-- 注入DataSource -->
    <property name="dataSource" ref="cPDataSource"></property>
</bean>
  1. 配置事務的增強
<tx:advice id="txadvice" transaction-manager="transactionManager">
    <!-- 事務操作 -->
    <tx:attributes>
        <!-- 
            配置匹配規則
                1. 直接設定方法名
                2. x+* 所有以x開頭的方法都執行事務
         -->
        <tx:method name="update*" propagation="REQUIRED"/>
    </tx:attributes>
</tx:advice>

說明: propagation="REQUIRED" 是設定隔離級別

  1. 配置切面
<aop:config>
    <!-- 切入點 -->
    <aop:pointcut expression="execution(* com.cfox.spring.UserService.*(..))" id="userServiceCut"/>
    <!-- 切面 -->
    <aop:advisor advice-ref="txadvice" pointcut-ref="userServiceCut"/>
</aop:config>

注意<aop:advisor advice-ref="txadvice" pointcut-ref="userServiceCut"/> 這一句,advice-ref 是設定的增強,pointcut-ref 是設定切入點

在Spring 中使用事務基本配置就這些。

下面是一個完整的例子的程式碼

在Spring 中使用c3p0 和 JdbcTemplate

<?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.xsd
        http://www.springframework.org/schema/context 
        http://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/tx 
        http://www.springframework.org/schema/tx/spring-tx.xsd
        http://www.springframework.org/schema/aop 
        http://www.springframework.org/schema/aop/spring-aop.xsd">

    <bean id="cPDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="com.mysql.jdbc.Driver" />
        <property name="jdbcUrl" value="jdbc:mysql://127.0.0.1:3306/spring_db?useUnicode=true&characterEncoding=UTF-8" />
        <property name="user" value="root" />
        <property name="password" value="root" />
    </bean>
    
    <!-- 1.配置事務管理器 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <!-- 注入DataSource -->
        <property name="dataSource" ref="cPDataSource"></property>
    </bean>
    <!-- 2.配置事務的增強  -->
    <tx:advice id="txadvice" transaction-manager="transactionManager">
        <!-- 事務操作 -->
        <tx:attributes>
            <!-- 
                配置匹配規則
                    1. 直接設定方法名
                    2. x+* 所有以x開頭的方法都執行事務
             -->
            <tx:method name="update*" propagation="REQUIRED"/>
        </tx:attributes>
    </tx:advice>
    
    <!-- 3. 配置切面 -->
    <aop:config>
        <!-- 切入點 -->
        <aop:pointcut expression="execution(* com.cfox.spring.UserService.*(..))" id="userServiceCut"/>
        <!-- 切面 -->
        <aop:advisor advice-ref="txadvice" pointcut-ref="userServiceCut"/>
    </aop:config>
    
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="cPDataSource" />
    </bean>
    
    <bean id="userService" class="com.cfox.spring.UserService">
        <property name="userDao" ref="userDao"/>
    </bean>
    
    <bean id="userDao" class="com.cfox.spring.UserDao">
        <property name="jdbcTemplate" ref="jdbcTemplate"></property>
    </bean>
</beans>

兩個bean 實現:

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

    public void updateMoney(String name, int money) {
        String sql = "UPDATE user_acc SET salary=salary+? WHERE user_name=?";
        int row = jdbcTemplate.update(sql, money, name);
        System.out.println(row);
    }

}
public class UserService {
    private UserDao userDao;
    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }

    public void updateUserMoney() {
        userDao.updateMoney("王五", 1000);
        int a = 10 / 0; // 使丟擲異常,讓事務執行回滾
        userDao.updateMoney("李四", -1000);
    }
}

相關文章