文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

@DSTransactional注解原理

2023-08-17 06:28

关注

一、AOP实现步骤

一句话:使用自定义注解(切点)+interceptor(增强Advice)构成织入。

1.定义注解 DSTransactional

代码如下(示例):

ipackage com.baomidou.dynamic.datasource.annotation;import java.lang.annotation.*;@Target({ElementType.TYPE, ElementType.METHOD})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface DSTransactional {}

2.定义拦截器(增强)DynamicLocalTransactionAdvisor

代码如下(示例):

package com.baomidou.dynamic.datasource.aop;import com.baomidou.dynamic.datasource.tx.ConnectionFactory;import com.baomidou.dynamic.datasource.tx.TransactionContext;import lombok.extern.slf4j.Slf4j;import org.aopalliance.intercept.MethodInterceptor;import org.aopalliance.intercept.MethodInvocation;import org.springframework.util.StringUtils;import java.util.UUID;@Slf4jpublic class DynamicLocalTransactionAdvisor implements MethodInterceptor {    @Override    public Object invoke(MethodInvocation methodInvocation) throws Throwable {        if (!StringUtils.isEmpty(TransactionContext.getXID())) {            return methodInvocation.proceed();        }        boolean state = true;        Object o;        String xid = UUID.randomUUID().toString();        TransactionContext.bind(xid);        try {            o = methodInvocation.proceed();        } catch (Exception e) {            state = false;            throw e;        } finally {            ConnectionFactory.notify(state);            TransactionContext.remove();        }        return o;    }}

3.AspectJ定义注解为切点,并配置织入(关键代码)

public class DynamicDataSourceAutoConfiguration implements InitializingBean {    @Role(value = BeanDefinition.ROLE_INFRASTRUCTURE)    @ConditionalOnProperty(prefix = DynamicDataSourceProperties.PREFIX, name = "seata", havingValue = "false", matchIfMissing = true)    @Bean    public Advisor dynamicTransactionAdvisor() {        AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut();        pointcut.setExpression("@annotation(com.baomidou.dynamic.datasource.annotation.DSTransactional)");        return new DefaultPointcutAdvisor(pointcut, new DynamicLocalTransactionAdvisor());    }}

二.connection代理实现

Spring自带事务@Transactional的实现在一个事务里,只能有一个数据库connection,在动态多数据源里的现象就是只有第一个数据源,后面切的都失效了。
所以要想实现动态多数据源下的统一提交和回滚,就不能用Spring自带的。

PS:在很多数据库相关项目里,connection这个词是有歧义的,可能有的含义包括:事务、会话、数据库连接。
spring自带事务明显默认了一个事务会话就是一个数据库链接这种老思想。

1.从增强实现开始

从DynamicLocalTransactionAdvisor增强的invoke方法来看具体逻辑:
首先用到了TransactionContext,一个基于ThreadLocal的账本,记录了当前事务的xid。看下面代码注释。

public class DynamicLocalTransactionAdvisor implements MethodInterceptor {    @Override    public Object invoke(MethodInvocation methodInvocation) throws Throwable {        // 1-1. 如果有xid,直接反射调用原方法,说明会话已经创建。        if (!StringUtils.isEmpty(TransactionContext.getXID())) {            return methodInvocation.proceed();        }        // 1-2. 如果没有xid,说明新会话,首先生成xid,绑到上下文上。        boolean state = true;        Object o;        String xid = UUID.randomUUID().toString();        TransactionContext.bind(xid);        try {            o = methodInvocation.proceed();        } catch (Exception e) {            // 1-3. 执行原方法,如果有异常,修改状态为false            state = false;            throw e;        } finally {            // 1-4. 调用会话的notify方法,处理状态            ConnectionFactory.notify(state);            // 1-5. 删除会话上下文            TransactionContext.remove();        }        return o;    }}

2. TransactionContext 事务上下文

public class TransactionContext {    //记录了当前事务的xid    private static final ThreadLocal CONTEXT_HOLDER = new ThreadLocal<>();        public static String getXID() {        String xid = CONTEXT_HOLDER.get();        if (!StringUtils.isEmpty(xid)) {            return xid;        }        return null;    }        public static String unbind(String xid) {        CONTEXT_HOLDER.remove();        return xid;    }        public static String bind(String xid) {        CONTEXT_HOLDER.set(xid);        return xid;    }        public static void remove() {        CONTEXT_HOLDER.remove();    }}

3. ConnectionFactory 会话工厂(代表会话)

再来看1-4用到的ConnectionFactory,也是一个基于ThreadLocal的账本,记录了该【会话】中用到的所有【连接】。

public class ConnectionFactory {    private static final ThreadLocal> CONNECTION_HOLDER =            new ThreadLocal>() {                @Override                protected Map initialValue() {                    return new ConcurrentHashMap<>();                }            };    // 3-1:将【数据库连接】存到会话的CONNECTION_HOLDER中    public static void putConnection(String ds, ConnectionProxy connection) {        Map concurrentHashMap = CONNECTION_HOLDER.get();        if (!concurrentHashMap.containsKey(ds)) {            try {                // 3-2:禁用了自动提交,相当于先执行数据库操作,但暂停了commit。等待3-4循环批量处理                // 注意,这个connection是proxy,也就是真正的数据库连接,不是会话                connection.setAutoCommit(false);            } catch (SQLException e) {                e.printStackTrace();            }            // 把新的连接放入Map            concurrentHashMap.put(ds, connection);        }    }    public static ConnectionProxy getConnection(String ds) {        return CONNECTION_HOLDER.get().get(ds);    }   // 3-3:前面1-4调用的方法:    public static void notify(Boolean state) {        try {            Map concurrentHashMap = CONNECTION_HOLDER.get();            for (ConnectionProxy connectionProxy : concurrentHashMap.values()) {                // 3-4:循环调用了所有数据库连接的notify方法。有一个false就都rollback了。                connectionProxy.notify(state);            }        } finally {            // 3-5:会话结束,删除数据库连接账本            CONNECTION_HOLDER.remove();        }    }}

4. ConnectionProxy 数据库连接代理

数据库连接

public class ConnectionProxy implements Connection {    private Connection connection;    private String ds;    public ConnectionProxy(Connection connection, String ds) {        this.connection = connection;        this.ds = ds;    }    // 4-1:前面303调用的方法    public void notify(Boolean commit) {        try {            if (commit) {                connection.commit();    // 状态为true,则提交            } else {                connection.rollback();   // 状态为false,则提交            }            connection.close();        } catch (Exception e) {            log.error(e.getLocalizedMessage(), e);        }    }    @Override    public void commit() throws SQLException {        // connection.commit();    }    // ....略}

5. AbstractRoutingDataSource 的改造:让动态数据库连接获取connection时登记到会话账本

public abstract class AbstractRoutingDataSource extends AbstractDataSource {    protected abstract DataSource determineDataSource();    @Override    public Connection getConnection() throws SQLException {        String xid = TransactionContext.getXID();        if (StringUtils.isEmpty(xid)) { // 非DSTransaction            return determineDataSource().getConnection();        } else {    // DSTransaction            String ds = DynamicDataSourceContextHolder.peek();            ConnectionProxy connection = ConnectionFactory.getConnection(ds);            return connection == null ? getConnectionProxy(ds, determineDataSource().getConnection()) : connection;        }    }    @Override    public Connection getConnection(String username, String password) throws SQLException {        String xid = TransactionContext.getXID();        if (StringUtils.isEmpty(xid)) { // 非DSTransaction            return determineDataSource().getConnection(username, password);        } else {  // DSTransaction            String ds = DynamicDataSourceContextHolder.peek();            ConnectionProxy connection = ConnectionFactory.getConnection(ds);            return connection == null ? getConnectionProxy(ds, determineDataSource().getConnection(username, password))                : connection;        }    }    private Connection getConnectionProxy(String ds, Connection connection) {        ConnectionProxy connectionProxy = new ConnectionProxy(connection, ds);        // 调用了3-1        ConnectionFactory.putConnection(ds, connectionProxy);        return connectionProxy;    }    // ... 略}

使用注意

不可与Transactional混用
2.目前只支持统一提交和回滚,更复杂的请用seata
3.3.4版本之前不加@DS注解会报错

来源地址:https://blog.csdn.net/wangchengqi1997/article/details/127821521

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯