中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

一文解析Spring編程式和聲明式事務實例講解

發布時間:2020-04-10 21:19:28 來源:網絡 閱讀:200 作者:wx5d9ed7c8443c3 欄目:編程語言

接上一篇:一文解析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

項目結構:

一文解析Spring編程式和聲明式事務實例講解

開發工具:

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;
    public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
        this.jdbcTemplate = jdbcTemplate;
    }
    // 對數據操作的方法不包含業務操作
    /**
     * 小王少錢的方法
     */
    public void reduceMoney() {
        String sql = "update account set salary=salary-? where username=?";
        jdbcTemplate.update(sql, 1000, "小王");
    }
    /**
     * 小馬多錢的方法
     */
    public void addMoney() {
        String sql = "update account set salary=salary+? where username=?";
        jdbcTemplate.update(sql, 1000, "小馬");
    }
}

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;
    public void setOrdersDao(OrdersDao ordersDao) {
        this.ordersDao = ordersDao;
    }
    // 調用dao的方法
    // 業務邏輯,寫轉賬業務
    public void accountMoney() {
        // 小馬多1000
        ordersDao.addMoney();
        // 加入出現異常如下面int i=10/0(銀行中可能為突然停電等。。。);結果:小馬賬戶多了1000而小王賬戶沒有少錢
        // 解決辦法是出現異常后進行事務回滾
        int i = 10 / 0;// 事務管理配置后異常已經解決
        // 小王 少1000
        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;
    public void setOrdersDao(OrdersDao ordersDao) {
        this.ordersDao = ordersDao;
    }
    // 調用dao的方法
    // 業務邏輯,寫轉賬業務
    public void accountMoney() {
        // 小馬多1000
        ordersDao.addMoney();
        // 加入出現異常如下面int i=10/0(銀行中可能為突然停電等。。。);結果:小馬賬戶多了1000而小王賬戶沒有少錢
        // 解決辦法是出現異常后進行事務回滾
        // int i = 10 / 0;// 事務管理配置后異常已經解決
        // 小王 少1000
        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>
向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

应用必备| 高邑县| 辉县市| 连山| 灵武市| 长子县| 专栏| 漯河市| 密山市| 达尔| 广昌县| 乌审旗| 滨海县| 辽阳县| 泾源县| 固原市| 运城市| 文水县| 丰原市| 涡阳县| 庆云县| 勃利县| 宁海县| 庆安县| 奈曼旗| 托克逊县| 永靖县| 波密县| 岑巩县| 吴川市| 灌南县| 乌拉特中旗| 通化县| 泸定县| 湘阴县| 双鸭山市| 高陵县| 青冈县| 云安县| 开鲁县| 普宁市|