从01开始 从01开始
首页
  • 计算机科学导论
  • 数字电路
  • 计算机组成原理

    • 计算机组成原理-北大网课
  • 操作系统
  • Linux
  • Docker
  • 计算机网络
  • 计算机常识
  • Git
  • JavaSE
  • Java高级
  • JavaEE

    • Ant
    • Maven
    • Log4j
    • Junit
    • JDBC
    • XML-JSON
  • JavaWeb

    • 服务器软件
    • Servlet
  • Spring
  • 主流框架

    • Redis
    • Mybatis
    • Lucene
    • Elasticsearch
    • RabbitMQ
    • MyCat
    • Lombok
  • SpringMVC
  • SpringBoot
  • 学习网课的心得
  • 输入法
  • 节假日TodoList
  • 其他
  • 关于本站
  • 网站日记
  • 友人帐
  • 如何搭建一个博客
GitHub (opens new window)

peterjxl

人生如逆旅,我亦是行人
首页
  • 计算机科学导论
  • 数字电路
  • 计算机组成原理

    • 计算机组成原理-北大网课
  • 操作系统
  • Linux
  • Docker
  • 计算机网络
  • 计算机常识
  • Git
  • JavaSE
  • Java高级
  • JavaEE

    • Ant
    • Maven
    • Log4j
    • Junit
    • JDBC
    • XML-JSON
  • JavaWeb

    • 服务器软件
    • Servlet
  • Spring
  • 主流框架

    • Redis
    • Mybatis
    • Lucene
    • Elasticsearch
    • RabbitMQ
    • MyCat
    • Lombok
  • SpringMVC
  • SpringBoot
  • 学习网课的心得
  • 输入法
  • 节假日TodoList
  • 其他
  • 关于本站
  • 网站日记
  • 友人帐
  • 如何搭建一个博客
GitHub (opens new window)
  • JavaSE

  • JavaSenior

  • JavaEE

  • JavaWeb

  • Spring

    • Spring介绍
    • 程序中的耦合
    • IoC的概念和作用
    • Spring中的依赖注入
    • 基于注解的IoC
    • 使用基于XML的IoC完成单表的CRUD
    • 使用基于注解的IoC完成单表的CRUD
    • IoC的纯注解配置
    • Spring整合Junit
    • 事务问题
      • 环境准备
      • 添加一个转账功能
      • 新建测试类
      • 如果有异常.....
      • 问题分析
      • connection工具类
      • 事务管理工具类
      • 改造service实现类
      • 改造dao实现类
      • 配置IoC
      • 测试
      • 总结
      • 源码
    • 代理模式
    • AOP的概念和入门
    • 基于注解的AOP
    • Spring的JdbcTemplate
    • JdBCDaoSupport
    • 基于XML的AOP实现事务控制
    • 基于注解的AOP实现事务控制
    • Spring的事务控制
    • 基于XML的声明式事务控制
    • 基于注解的声明式事务控制
    • 纯注解实现事务控制
    • Spring编程式事务控制
    • Spring5新特性
    • Java
  • 主流框架

  • SpringMVC

  • SpringBoot

  • Java并发

  • Java源码

  • JVM

  • 韩顺平

  • Java
  • Java
  • Spring
2023-05-08
目录

事务问题

# 70.事务问题

在讲解AOP之前,我们先看看我们之前的案例中有什么问题,然后再引出AOP   ‍

‍

# 环境准备

为了方便演示,我们在之前demo5的分支上继续开发;

‍

‍

‍

# 添加一个转账功能

我们添加一个转账的功能。首先在接口层​IAccountService​添加一个transfer方法:

  /**
     * 转账
     * @param sourceName 转出账户名称
     * @param targetName 转入账户名称
     * @param money 转账金额
     */
    void transfer(String sourceName, String targetName, Float money);
1
2
3
4
5
6
7

‍

‍

然后我们需要在dao接口​IAccountDao​增加一个根据名称查找账户的方法:

/**
     * 根据名称查询账户
     * @param accountName
     * @return 如果有唯一的结果就返回,如果没有结果就返回 null
     * 如果结果集超过一个就抛异常
     */
    Account findAccountByName(String accountName);
1
2
3
4
5
6
7

‍

然后我们在​AccountDaoImpl​实现这个方法:

@Override
    public Account findAccountByName(String accountName) {
        try {
            List<Account> accounts = runner.query("select * from account where name = ? ", new BeanListHandler<Account>(Account.class), accountName);
            if (accounts == null || accounts.size() == 0) {
                return null;
            }
            if (accounts.size() > 1) {
                throw new RuntimeException("结果集不唯一,数据有问题");
            }
            return accounts.get(0);
        } catch (SQLException e) {
            throw new RuntimeException(e);
        }
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

‍

‍

‍

然后我们在​AccountServiceImpl​实现类里实现这个方法。步骤如下:

  1. 根据名称查询转出账户
  2. 根据名称查询转入账户
  3. 转出账户减钱
  4. 转入账户加钱
  5. 更新转出账户
  6. 更新转入账户

‍

@Override
public void transfer(String sourceName, String targetName, Float money) {
    // 1. 根据名称查询转出账户
    Account source = accountDao.findAccountByName(sourceName);
    // 2. 根据名称查询转入账户
    Account target = accountDao.findAccountByName(targetName);
    // 3. 转出账户减钱
    source.setMoney(source.getMoney() - money);
    // 4. 转入账户加钱
    target.setMoney(target.getMoney() + money);
    // 5. 更新转出账户
    accountDao.updateAccount(source);
    // 6. 更新转入账户
    accountDao.updateAccount(target);
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

‍

‍

# 新建测试类

我们测试下这个方法,用aaa账户给bbb账户转100块钱:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:bean.xml")
public class AccountServiceTest {

    @Autowired
    private IAccountService as;

    @Test
    public  void testTransfer(){
        as.transfer("aaa","bbb",100f);
    }
}
1
2
3
4
5
6
7
8
9
10
11
12

‍

运行前,两个整合都是1000块:

id	name	money
1	aaa	1000
2	bbb	1000
3	ccc	1000
1
2
3
4

‍

‍

运行后,成功转账:

id	name	money
1	aaa	900
2	bbb	1100
3	ccc	1000
1
2
3
4

‍

# 如果有异常.....

如果在转账的过程中,发生了异常,怎么办呢?我们可以测试下,自己创造一个异常:

 @Override
    public void transfer(String sourceName, String targetName, Float money) {
        // 1. 根据名称查询转出账户
        Account source = accountDao.findAccountByName(sourceName);
        // 2. 根据名称查询转入账户
        Account target = accountDao.findAccountByName(targetName);
        // 3. 转出账户减钱
        source.setMoney(source.getMoney() - money);
        // 4. 转入账户加钱
        target.setMoney(target.getMoney() + money);
        // 5. 更新转出账户
        accountDao.updateAccount(source);

        int i = 1/0;

        // 6. 更新转入账户
        accountDao.updateAccount(target);
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

‍

此时运行结果,确实抛出了异常;现在关键的问题是:数据库里的金额不对,aaa账户的100块钱消失了!

java.lang.ArithmeticException: / by zero


id	name	money
1	aaa	800
2	bbb	1100
3	ccc	1000
1
2
3
4
5
6
7

‍

‍

‍

# 问题分析

上述问题的产生,是因为我们没有使用开启事务,因此会自动提交,不会回滚。

在转账事务,获取了多次连接对象:查询a账户一次,查询b账户一次,更新a账户一次,更新b账户一次,发送了多次请求,每次请求成功后就会自动提交。如果中途出错,则后面的代码不会执行并提交,但之前的已经提交了

所以,转账事务里的操作,应该都是同一个connection操作,在业务层控制事务。我们可以使用ThreadLocal对象把Connection和当前线程绑定,从而使个线程中只有一个能控制事务的对象。

‍

‍

# connection工具类

我们可以写个工具类,用来获取connection,并新增一个dataSource属性和set方法用于注入

package com.peterjxl.utils;

import javax.sql.DataSource;
import java.sql.Connection;

/**
 * 连接的工具类,它用于从数据源中获取一个连接,并且实现和线程的绑定
 */
public class ConnectonUtils {

    private ThreadLocal<Connection> tl = new ThreadLocal<Connection>();

    private DataSource dataSource;

    public void setDataSource(DataSource dataSource) {
        this.dataSource = dataSource;
    }

    /**
     * 获取当前线程上的连接
     * @return
     */
    public Connection getThreadConnection() {
        try {
            // 1. 先从 ThreadLocal 上获取
            Connection conn = tl.get();

            // 2. 判断当前线程上是否有连接
            if (conn == null) {
                // 3. 从数据源中获取一个连接,并且存入 ThreadLocal 中
                conn = dataSource.getConnection();
                tl.set(conn);
            }
            // 4. 返回当前线程上的连接
            return conn;
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
  
    /**
     * 把连接和线程解绑
     */
    public void removeConnection() {
        tl.remove();
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47

‍

‍

# 事务管理工具类

再写一个工具类,管理事务:

package com.peterjxl.utils;

/**
 * 和事务管理相关的工具类,它包含了开启事务,提交事务,回滚事务和释放连接
 */
public class TransactionManager {

    private ConnectonUtils connectonUtils;

    public void setConnectonUtils(ConnectonUtils connectonUtils) {
        this.connectonUtils = connectonUtils;
    }

    /**
     * 开启事务
     */
    public void beginTransaction() {
        try {
            connectonUtils.getThreadConnection().setAutoCommit(false);
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * 提交事务
     */
    public void commit() {
        try {
            connectonUtils.getThreadConnection().commit();
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * 回滚事务
     */
    public void rollback() {
        try {
            connectonUtils.getThreadConnection().rollback();
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

    /**
     * 释放连接
     */
    public void release() {
        try {
            connectonUtils.getThreadConnection().close(); // 还回连接池中
            connectonUtils.removeConnection();
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59

‍

注意Tomcat等Web服务器,也会有线程池技术,当不再使用这个线程时,需要将connection移除;否则下次再获取这个线程时,还是能判断出是该线程有connection的,所以我们需要解绑。

‍

‍

# 改造service实现类

现在我们可以给service实现类加上TransactionManager​成员变量和set方法,用来注入;

然后给每个方法,都加上事务控制,由于方法有很多,我们就不全部贴出来了:

@Override
public List<Account> findAllAccount() {

    try {
        // 1. 开启事务
        txManager.beginTransaction();
        // 2. 执行操作
        List<Account> accounts = accountDao.findAllAccount();
        //3. 提交事务
        txManager.commit();
        //4. 返回结果
        return accounts;
    }catch (Exception e) {
        // 5. 回滚操作
        txManager.rollback();
        throw new RuntimeException(e);
    }finally {
        // 6. 释放连接
        txManager.release();
    }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

‍

也就是每个方法中,都有重复的代码块:开启事务、提交事务、返回结果、异常处理等代码。

‍

‍

# 改造dao实现类

由于需要使用当前线程上的连接,因此我们需要引入ConnectionUtils​的依赖,并且在Queryrunner中传入当前线程:

@Repository("accountDao")
public class AccountDaoImpl implements IAccountDao {

    private QueryRunner runner;

    private ConnectionUtils connectionUtils;

    public void setRunner(QueryRunner runner) {
        this.runner = runner;
    }

    public void setConnectionUtils(ConnectionUtils connectionUtils) {
        this.connectionUtils = connectionUtils;
    }

    @Override
    public List<Account> findAllAccount() {
        try {
            return runner.query(connectionUtils.getThreadConnection(), "select * from account", new BeanListHandler<Account>(Account.class));
        } catch (SQLException e) {
            throw new RuntimeException(e);
        }
    }
  
    // .........
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

‍

也就是说,每个方法的query方法里,都要传入connectionUtils.getThreadConnection()​这个对象

‍

‍

# 配置IoC

接下来就是配置注入了

  1. 给service层 注入事务管理工具类
  2. 给dao层 注入connectionUtils​工具类
  3. ​QueryRunner​不再需要dataSource​注入,而是connectionUtils​需要注入dataSource​
  4. 给事务管理工具类注入​connectionUtils​

‍

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!-- 配置Service -->
    <bean id="accountService" class="com.peterjxl.service.impl.AccountServiceImpl">
        <!-- 注入dao -->
        <property name="accountDao" ref="accountDao"/>
        <property name="txManager" ref="txManager"/>
    </bean>

    <!--配置Dao对象-->
    <bean id="accountDao" class="com.peterjxl.dao.impl.AccountDaoImpl">
        <!-- 注入QueryRunner -->
        <property name="runner" ref="runner"/>
        <!-- 注入ConnectionUtils -->
        <property name="connectionUtils" ref="connectionUtils"/>
    </bean>

    <!--配置QueryRunner-->
    <bean id="runner" class="org.apache.commons.dbutils.QueryRunner" scope="prototype"/>

    <!-- 配置数据源 -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <!--连接数据库的必备信息-->
        <property name="driverClass" value="com.mysql.cj.jdbc.Driver"/>
        <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/learnSpring"/>
        <property name="user" value="learnSpringUser"/>
        <property name="password" value="learnSpringPassword"/>
    </bean>

    <!-- 配置Connection的工具类 ConnectionUtils -->
    <bean id="connectionUtils" class="com.peterjxl.utils.ConnectionUtils">
        <!-- 注入数据源-->
        <property name="dataSource" ref="dataSource"/>
    </bean>

    <!-- 配置事务管理器-->
    <bean id="txManager" class="com.peterjxl.utils.TransactionManager">
        <!-- 注入ConnectionUtils -->
        <property name="connectionUtils" ref="connectionUtils"/>
    </bean>
</beans>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45

‍

‍

# 测试

此时我们再次测试转账,可以看到即使发生了异常,也能正常回滚,也就是成功使用了事务!

‍

‍

# 总结

虽然我们已经实现了事务,但目前项目中仍存在不少问题:

  1. 配置非常麻烦,很多依赖注入,引入了事务管理工具类和connection工具类
  2. 不仅仅引入了类之间的依赖,还有方法之间的依赖, 例如事务工具类有个方法名改了,所有service类都得跟着改
  3. 有很多的重复代码

有什么解决办法吗?有的,使用代理,增强我们的方法!下一篇博客就会回顾代理模式,其实之前已经讲过一些基本的概念并演示了,例如:JavaWeb-Filter案例 (opens new window)

‍

# 源码

本项目已将源码上传到GitHub (opens new window)和Gitee (opens new window)上。并且创建了分支demo9,读者可以通过切换分支来查看本文的示例代码。

在GitHub上编辑此页 (opens new window)
上次更新: 2023/6/7 11:49:40
Spring整合Junit
代理模式

← Spring整合Junit 代理模式→

Theme by Vdoing | Copyright © 2022-2023 粤ICP备2022067627号-1 粤公网安备 44011302003646号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式