Java高频面试题:MyBatis如何实现动态数据源切换?

大家好,我是锋哥。今天分享关于【Java高频面试题:MyBatis如何实现动态数据源切换?】面试题 。希望对大家有帮助;

Java高频面试题:MyBatis如何实现动态数据源切换?

1️⃣ 基本思路

在 MyBatis(或 Spring + MyBatis)中,动态切换数据源的核心思想是:

  1. 定义多个数据源(DataSource)。
  2. 使用一个 动态数据源路由器(DynamicDataSource),在运行时根据上下文决定使用哪个数据源。
  3. 在方法或线程执行前,设置当前线程使用的数据源标识(通常用 ThreadLocal 保存)。
  4. MyBatis 的 SqlSession 或 Spring JdbcTemplate 会通过动态数据源路由器获取当前的数据源。

关键点:线程安全。每个线程单独维护自己的数据源标识,防止并发下数据源错乱。


2️⃣ 核心实现步骤

2.1 定义数据源

通常在 application.ymlapplication.properties 中配置多个数据源,例如:

复制代码
spring:
  datasource:
    master:
      url: jdbc:mysql://localhost:3306/master
      username: root
      password: root
      driver-class-name: com.mysql.cj.jdbc.Driver
    slave:
      url: jdbc:mysql://localhost:3306/slave
      username: root
      password: root
      driver-class-name: com.mysql.cj.jdbc.Driver

在 Spring 中配置对应的 DataSource Bean:

复制代码
@Bean
@Primary
@ConfigurationProperties(prefix = "spring.datasource.master")
public DataSource masterDataSource() {
    return DataSourceBuilder.create().build();
}

@Bean
@ConfigurationProperties(prefix = "spring.datasource.slave")
public DataSource slaveDataSource() {
    return DataSourceBuilder.create().build();
}

2.2 创建动态数据源

复制代码
public class DynamicDataSource extends AbstractRoutingDataSource {

    // 核心方法,根据当前线程的数据源标识返回对应 DataSource
    @Override
    protected Object determineCurrentLookupKey() {
        return DataSourceContextHolder.getDataSource();
    }
}

DataSourceContextHolder 用于管理线程数据源:

复制代码
public class DataSourceContextHolder {
    private static final ThreadLocal<String> CONTEXT = new ThreadLocal<>();

    public static void setDataSource(String dataSource) {
        CONTEXT.set(dataSource);
    }

    public static String getDataSource() {
        return CONTEXT.get();
    }

    public static void clear() {
        CONTEXT.remove();
    }
}

然后在配置类里:

复制代码
@Bean
public DataSource dynamicDataSource(DataSource masterDataSource, DataSource slaveDataSource) {
    Map<Object, Object> targetDataSources = new HashMap<>();
    targetDataSources.put("master", masterDataSource);
    targetDataSources.put("slave", slaveDataSource);

    DynamicDataSource dynamicDataSource = new DynamicDataSource();
    dynamicDataSource.setDefaultTargetDataSource(masterDataSource); // 默认数据源
    dynamicDataSource.setTargetDataSources(targetDataSources);
    return dynamicDataSource;
}

2.3 MyBatis 配置动态数据源

在 Spring Boot 中,通常直接配置 MyBatis 使用动态数据源:

复制代码
@Bean
public SqlSessionFactory sqlSessionFactory(DataSource dynamicDataSource) throws Exception {
    SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();
    sessionFactory.setDataSource(dynamicDataSource);
    return sessionFactory.getObject();
}

2.4 在代码中切换数据源

可以在服务方法上通过 AOP 或注解 来切换数据源,例如:

复制代码
@Target({ ElementType.METHOD, ElementType.TYPE })
@Retention(RetentionPolicy.RUNTIME)
public @interface DataSource {
    String value() default "master";
}

AOP 切面实现:

复制代码
@Aspect
@Component
public class DataSourceAspect {

    @Before("@annotation(dataSource)")
    public void switchDataSource(DataSource dataSource) {
        DataSourceContextHolder.setDataSource(dataSource.value());
    }

    @After("@annotation(dataSource)")
    public void clearDataSource(DataSource dataSource) {
        DataSourceContextHolder.clear();
    }
}

然后在 Service 方法上标注:

复制代码
@Service
public class UserService {

    @DataSource("slave")
    public List<User> getUsersFromSlave() {
        return userMapper.selectAll();
    }

    @DataSource("master")
    public void insertUser(User user) {
        userMapper.insert(user);
    }
}

✅ 这样就实现了 按方法动态切换数据源


3️⃣ 注意事项

  1. 事务管理

    动态数据源切换必须配合 Spring 的事务管理,否则可能出现事务跨数据源异常。

    • 通常使用 @Transactional 注解配合 AbstractRoutingDataSource 即可。
  2. 线程安全

    必须使用 ThreadLocal 来保证每个线程独立使用数据源标识,否则并发请求可能混乱。

  3. 性能考虑

    切换数据源比单一数据源略有开销,尤其是在高并发下,需要注意连接池的配置。


💡 总结:

MyBatis 的动态数据源切换本质是 AbstractRoutingDataSource + ThreadLocal 的组合,通过 AOP 或手动设置当前线程的数据源来实现方法级的数据源切换。

相关推荐
kkeeper~2 小时前
0基础C语言积跬步之深入理解指针(5下)
c语言·开发语言
一直不明飞行2 小时前
Java的equals(),hashCode()应该在什么时候重写
java·开发语言·jvm
REDcker2 小时前
有限状态机与状态模式详解 FSM建模Java状态模式与C++表驱动模板实践
java·c++·状态模式
盲敲代码的阿豪2 小时前
Python 入门基础教程(爬虫前置版)
开发语言·爬虫·python
你的保护色3 小时前
【无标题】
java·服务器·网络
basketball6163 小时前
C++ 构造函数完全指南:从入门到进阶
java·开发语言·c++
互联科技报3 小时前
2026超融合选型:Top5品牌与市场格局解读
开发语言·perl
weixin199701080163 小时前
[特殊字符] 智能数据采集:数字化转型的“数据石油勘探队”(附Python实战源码)
开发语言·python
淘矿人3 小时前
Claude辅助DevOps实践
java·大数据·运维·人工智能·算法·bug·devops
想唱rap4 小时前
IO多路转接之poll
服务器·开发语言·数据库·c++