久久精品精选,精品九九视频,www久久只有这里有精品,亚洲熟女乱色综合一区
    分享

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

     Wlfnx4dnqz6ew3 2019-05-23

    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包地址:鏈接:https://pan.baidu.com/s/1tqy-mVKxSutsIIvYgtC3Rw 密碼: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;
    
        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./schema/beans"
        xmlns:xsi="http://www./2001/XMLSchema-instance" xmlns:context="http://www./schema/context"
        xmlns:aop="http://www./schema/aop" xmlns:tx="http://www./schema/tx"
        xsi:schemaLocation="http://www./schema/beans http://www./schema/beans/spring-beans-2.5.xsd  
    http://www./schema/context http://www./schema/context/spring-context-2.5.xsd  
    http://www./schema/aop http://www./schema/aop/spring-aop-2.5.xsd  
    http://www./schema/tx http://www./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>

      本站是提供個人知識管理的網絡存儲空間,所有內容均由用戶發布,不代表本站觀點。請注意甄別內容中的聯系方式、誘導購買等信息,謹防詐騙。如發現有害或侵權內容,請點擊一鍵舉報。
      轉藏 分享 獻花(0

      0條評論

      發表

      請遵守用戶 評論公約

      類似文章 更多

      主站蜘蛛池模板: 国产午夜亚洲精品国产成人| 久久精品一本到99热免费| 久久综合亚洲色HEZYO国产| 无码精品久久久久久人妻中字 | 色噜噜噜亚洲男人的天堂| 老司机导航亚洲精品导航| 宅男在线永久免费观看网| 一区二区亚洲人妻精品| 无码人妻人妻经典| 久久夜色精品国产嚕嚕亚洲AV| 亚洲欧美日韩在线码| 免费无码又爽又刺激网站| 色综合久久久无码中文字幕| 无码专区 人妻系列 在线| 天干天干天啪啪夜爽爽AV| 无码日韩精品一区二区三区免费 | 成人网站WWW污污污网站| 亚洲欧美日韩在线码| 韩国无码AV片在线观看网站| 国产精品有码在线观看| 少妇愉情理伦片BD| 日韩加勒比一本无码精品| 国产第一页浮力影院草草影视 | 国精品无码一区二区三区左线| 深夜视频国产在线观看| 亚洲综合小说另类图片五月天| 欧美性群另类交| 精品剧情V国产在线观看| 无码囯产精品一区二区免费| 影音先锋2020色资源网| 肥臀浪妇太爽了快点再快点 | 欧洲人妻丰满AV无码久久不卡| 国产亚洲精品AA片在线播放天| 国产69精品久久久久777 | 亚洲色大成成人网站久久| 成人久久免费网站| 国产L精品国产亚洲区在线观看| 2021亚洲国产精品无码| 亚洲精品国产精品乱码视色| 国产揄拍国产精品| 男女扒开双腿猛进入爽爽免费看 |