多线程下,解决银行转账,事务操作的原子性

多线程下,解决银行转账,不同线程对事务操作的原子性:

用到了ThrealLocal类,来实现。
ThrealLocal中是map集合,存储的是map(thread,T):键标识是当前线程,T标识泛型值。功能:能够唯一标识一个线程,不同线程可以有不同标识。这样可以保证在转账过程中,只能在同一线程中进行事务的操作。保证不同线程之间,事务的操作是隔离的

定义工具类:

public class DbUtilsDataSource {

    private static DruidDataSource druidDataSource= null;

    public  static  ThreadLocal<Connection> threadLocal=new ThreadLocal<>();

    /*放入静态代码块中,在调用静态方法getDataResource的时候触发类加载*/
    static {
        try {
            /*从配置文件中获取*/
            Properties properties = new Properties();
            /*获取配置文件流对象*/
            InputStream is = DbUtilsDataSource.class.getClassLoader().getResourceAsStream("com/xzq/druidutils/druidmess.properties");
            /*加载配置文件*/
            properties.load(is);
            is.close();
            /*利用工厂方式获取一个连接池对象*/
            druidDataSource = (DruidDataSource) DruidDataSourceFactory.createDataSource(properties);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static Connection  getConnection() {

        /*刚开始threadlocal中是没有值的,为null,所以*/
        Connection connection = threadLocal.get();

        if(connection ==null)
        {
            try {
                //获取连接池对象,就当如到threadlical对象中。
                connection = getDataResource().getConnection();
                /*把connection放入threadlocal*/
                threadLocal.set(connection);
            } catch (SQLException e) {
                e.printStackTrace();
            }
        }
        return connection;

    }
    public static  DataSource getDataResource(){

        return  druidDataSource;
    }

    /*开启事务*/
    public static void start() throws SQLException {
        Connection connection = getConnection();
        if(connection!=null){
          connection.setAutoCommit(false);
        }
    }

    /*提交事务*/
    public static void  commit() throws SQLException {
        Connection connection =  getConnection();
        if(connection!=null){
            connection.commit();
        }
    }

    /*回滚事务*/
    public  static void rollback() throws SQLException {
        Connection connection = getConnection();
        if (connection != null) {
          connection.rollback();
        }
    }

        /*当前线程,事务操作完成,关闭连接,解绑(也就是将threadlocal中的值移除掉)*/
        public static void close() throws SQLException {
            Connection connection = getConnection();
            if (connection != null) {
                connection.close();
                threadLocal.remove();
            }
        }

    }

转账:

 public void sendmoney(String money,String fcard,String tcard ) {

        /*获取连接*/
        Connection connection = DbUtilsDataSource.getConnection();

        try {
            /*开启事务*/
            DbUtilsDataSource.start();
            td.addMoney(money,tcard);
            td.subtracMoney(money,fcard);
            //提交事务
            DbUtilsDataSource.commit();
        } catch (Exception e) {
            try {
                /*事务回滚*/
                DbUtilsDataSource.rollback();
            } catch (SQLException e1) {
                e1.printStackTrace();
            }
        }finally {
            try {
                DbUtilsDataSource.close();
            } catch (SQLException e) {
                e.printStackTrace();
            }

        }
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
多线程异步操作回滚事务可以通过使用`@Transactional`注解和`TransactionSynchronizationManager`类来实现。具体步骤如下: 1.在异步方法上添加`@Transactional(rollbackFor = Exception.class)`注解,表示该方法需要进行事务管理,并且在出现异常时需要回滚事务。 2.在同步方法中调用异步方法时,使用`TransactionSynchronizationManager.registerSynchronization()`方法注册一个事务同步器,该同步器会在事务提交或回滚时被调用。 3.在事务同步器中调用`TransactionSynchronizationManager.isActualTransactionActive()`方法判断当前是否有活动的事务,如果有,则调用`TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()`方法将事务标记为回滚状态。 下面是一个示例代码: ```java @Service public class MyService { @Autowired private MyAsyncService myAsyncService; @Transactional(rollbackFor = Exception.class) public void doSomething() { // 业务代码... try { // 数据库操作... } catch (Exception e) { // 调用异步方法完成数据库操作 myAsyncService.doSomethingAsync(); // 注册一个事务同步器 TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronizationAdapter() { @Override public void afterCompletion(int status) { // 判断当前是否有活动的事务 if (TransactionSynchronizationManager.isActualTransactionActive()) { // 将事务标记为回滚状态 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); } } }); } } } @Service public class MyAsyncService { @Async @Transactional(rollbackFor = Exception.class) public void doSomethingAsync() { // 异步方法业务代码... try { // 异步方法数据库操作... } catch (Exception e) { // 抛出异常,事务会自动回滚 throw new RuntimeException(e); } } } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值