• 9.4JavaEE——声明式事务管理(一)基于XML方式的声明式事务


    一、如何实现XML方式的声明式事务

            基于XML方式的声明式事务管理是通过在配置文件中配置事务规则的相关声明来实现的。在使用XML文件配置声明式事务管理时,首先要引入tx命名空间,在引入tx命名空间之后,可以使用元素来配置事务管理的通知,进而通过Spring AOP实现事务管理。

           配置元素时,通常需要指定 id 和 transaction-manager 属性,其中,id 属性是配置文件中的唯一标识,transaction-manager 属性用于指定事务管理器。除此之外,元素还包含子元素元素可配置多个子元素,子元素主要用于配置事务的属性。

    二、元素的常用属性     

    属性说明
    name用于指定方法名的匹配模式,该属性为必选属性,它指定了与事务属性相关的方法名。
    propagation用于指定事务的传播行为。
    isolation用于指定事务的隔离级别。
    read-only用于指定事务是否只读。
    timeout用于指定事务超时时间。
    rollback-for用于指定触发事务回滚的异常类。
    no-rollback-for用于指定不触发事务回滚的异常类。

            接下来通过一个案例演示如何通过XML方式实现Spring的声明式事务管理。本案例以9.2小节的项目代码和数据表为基础,编写一个模拟银行转账的程序,要求在转账时通过Spring对事务进行控制。案例具体实现步骤如下。

    1、导入依赖

            在chapter09项目的pom.xml文件中加入aspectjweaver依赖包和aopalliance依赖包作为实现切面所需的依赖包。

    1. org.aspectj
    2. aspectjweaver
    3. 1.9.6
    4. runtime
    5. aopalliance
    6. aopalliance
    7. 1.0

    2、定义Dao层方法

            AccountDao接口中声明转账方法transfer()。

    1. // 转账
    2. public void transfer(String outUser,
    3. String inUser,
    4. Double money);

    3、实现Dao层方法

            AccountDaoImpl实现类中实现AccountDao接口中的transfer()方法。

    1. // 转账 inUser:收款人; outUser:汇款人; money:收款金额
    2. public void transfer(String outUser, String inUser, Double money) {
    3. // 收款时,收款用户的余额=现有余额+所汇金额
    4. this.jdbcTemplate.update("update account set balance = balance +? "
    5. + "where username = ?",money, inUser);
    6. // 模拟系统运行时的突发性问题
    7. int i = 1/0;
    8. // 汇款时,汇款用户的余额=现有余额-所汇金额
    9. this.jdbcTemplate.update("update account set balance = balance-? "
    10. + "where username = ?",money, outUser);
    11. }

    4、修改配置文件

            修改chapter09项目的配置文件applicationContext.xml,添加命名空间等相关配置代码。

    1. "1.0" encoding="UTF-8"?>
    2. "http://www.springframework.org/schema/beans"
    3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    4. xmlns:aop="http://www.springframework.org/schema/aop"
    5. xmlns:tx="http://www.springframework.org/schema/tx"
    6. xsi:schemaLocation="http://www.springframework.org/schema/beans
    7. http://www.springframework.org/schema/beans/spring-beans.xsd
    8. http://www.springframework.org/schema/tx
    9. http://www.springframework.org/schema/tx/spring-tx.xsd
    10. http://www.springframework.org/schema/aop
    11. http://www.springframework.org/schema/aop/spring-aop.xsd">
    12. "dataSource" class=
    13. "org.springframework.jdbc.datasource.DriverManagerDataSource">
    14. "driverClassName" value="com.mysql.cj.jdbc.Driver" />
    15. "url" value="jdbc:mysql://localhost/spring?useUnicode=true&characterEncoding=utf-8&serverTimezone=Asia/Shanghai" />
    16. "username" value="root" />
    17. "password" value="root" />
    18. "jdbcTemplate"
    19. class="org.springframework.jdbc.core.JdbcTemplate">
    20. "dataSource" ref="dataSource" />
    21. "accountDao" class="com.itheima.AccountDaoImpl">
    22. "jdbcTemplate" ref="jdbcTemplate" />
    23. "transactionManager" class=
    24. "org.springframework.jdbc.datasource.DataSourceTransactionManager">
    25. "dataSource" ref="dataSource" />
    26. "txAdvice" transaction-manager="transactionManager">
    27. "*" propagation="REQUIRED"
    28. isolation="DEFAULT" read-only="false" />
    29. "execution(* com.itheima.*.*(..))"
    30. id="txPointCut" />
    31. "txAdvice" pointcut-ref="txPointCut" />

    5、测试系统

            创建测试类TransactionTest。

    1. public class TransactionTest {
    2. public static void main(String[] args) {
    3. ApplicationContext applicationContext =
    4. new ClassPathXmlApplicationContext("applicationContext.xml");
    5. // 获取AccountDao实例
    6. AccountDao accountDao =
    7. (AccountDao)applicationContext.getBean("accountDao");
    8. // 调用实例中的转账方法
    9. accountDao.transfer("lisi", "zhangsan", 100.0);
    10. // 输出提示信息
    11. System.out.println("转账成功!");}}

    6、查看表数据

            在执行转账操作前,也就是在执行第5不操作前,先查看account表中的数据。

    1. mysql> use spring
    2. Database changed
    3. mysql> select * from account;
    4. +----+----------+---------+
    5. | id | username | balance |
    6. +----+----------+---------+
    7. | 1 | zhangsan | 100 |
    8. | 3 | lisi | 500 |
    9. | 4 | wnagwu | 300 |
    10. +----+----------+---------+
    11. 3 rows in set (0.00 sec)

    7、再查看表数据

            在执行第5步后,控制台中报出了“/by zero”的算术异常信息。此时再次查询数据表account。

    1. mysql> select * from account;
    2. +----+----------+---------+
    3. | id | username | balance |
    4. +----+----------+---------+
    5. | 1 | zhangsan | 100 |
    6. | 3 | lisi | 500 |
    7. | 4 | wnagwu | 300 |
    8. +----+----------+---------+
    9. 3 rows in set (0.00 sec)

    未使用事物管理的缺陷

            由XML方式实现声明式事务管理的案例可知,zhangsan的账户余额增加了100,而lisi的账户确没有任何变化,这样的情况显然是不合理的。这就是没有事务管理,系统无法保证数据的安全性与一致性,下面使用事务管理解决该问题。

    8、使用事务管理测试系统

            在第4步的代码文件中添加事务管理的配置。 

    1. "txAdvice" transaction-manager="transactionManager">
    2. "*" propagation="REQUIRED"
    3. isolation="DEFAULT" read-only="false" />
    4. "execution(* com.itheima.*.*(..))"
    5. id="txPointCut" />
    6. "txAdvice" pointcut-ref="txPointCut"

  • 相关阅读:
    Docker与Kubernetes介绍
    【国庆活动】掌握这些SQL核心理念,一起轻松玩转国庆假期
    python 微信小程序 英语单词小程序代码分享
    VS_QT_4_Qt设计师
    【Android入门】6、ContentProvider:跨程序的数据共享:访问其他 App、被其他 App 访问
    147. 对链表进行插入排序 ●●
    yolov5训练问题
    YOLOv5改进之二:添加CBAM注意力机制
    XShell远程连接Ubuntu
    【allegro 17.4软件操作保姆级教程四】布线前准备之叠层和阻抗设计
  • 原文地址:https://blog.csdn.net/W_Fe5/article/details/139370009