欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring通過攔截器實(shí)現(xiàn)多數(shù)據(jù)源切換的示例代碼

 更新時(shí)間:2025年08月20日 10:02:16   作者:南姜先生  
本文主要介紹了Spring攔截器實(shí)現(xiàn)多數(shù)據(jù)源動(dòng)態(tài)切換,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

在 Spring 應(yīng)用中,特別是使用了多數(shù)據(jù)源的場(chǎng)景下,可以通過**攔截器(Interceptor)**機(jī)制來實(shí)現(xiàn)動(dòng)態(tài)數(shù)據(jù)源切換。Spring 提供了多種方式來管理多個(gè)數(shù)據(jù)源,并允許開發(fā)者根據(jù)業(yè)務(wù)邏輯的需求靈活地選擇不同的數(shù)據(jù)源進(jìn)行數(shù)據(jù)庫(kù)操作。

下面介紹如何通過攔截器實(shí)現(xiàn)多數(shù)據(jù)源切換的基本思路和具體步驟。

?? 一句話總結(jié):

? 通過自定義 HandlerInterceptor 或 MyBatis 的 Interceptor 攔截請(qǐng)求或 SQL 執(zhí)行過程,在合適的時(shí)機(jī)動(dòng)態(tài)設(shè)置當(dāng)前線程的數(shù)據(jù)源,從而實(shí)現(xiàn)基于請(qǐng)求或方法級(jí)別的多數(shù)據(jù)源切換。

?? 一、基本思路

  1. 配置多個(gè)數(shù)據(jù)源:首先需要在 Spring 配置類中定義多個(gè) DataSource Bean。
  2. 動(dòng)態(tài)數(shù)據(jù)源路由:創(chuàng)建一個(gè)代理數(shù)據(jù)源,它能夠根據(jù)某些條件(如當(dāng)前線程中的某個(gè)標(biāo)識(shí)符)選擇實(shí)際使用的數(shù)據(jù)源。
  3. 攔截請(qǐng)求或方法調(diào)用:使用 Spring MVC 的 HandlerInterceptor 或 MyBatis 的 Interceptor 來攔截請(qǐng)求或方法調(diào)用,在執(zhí)行之前設(shè)置當(dāng)前線程的數(shù)據(jù)源標(biāo)識(shí)符。
  4. 清除上下文:確保在請(qǐng)求結(jié)束后清理線程局部變量,避免影響后續(xù)請(qǐng)求。

??? 二、具體實(shí)現(xiàn)步驟

1.定義多個(gè)數(shù)據(jù)源

@Configuration
public class DataSourceConfig {

    @Bean(name = "primaryDataSource")
    @Primary
    @ConfigurationProperties(prefix = "spring.datasource.primary")
    public DataSource primaryDataSource() {
        return DataSourceBuilder.create().build();
    }

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

2.創(chuàng)建動(dòng)態(tài)數(shù)據(jù)源路由

public class DynamicDataSource extends AbstractRoutingDataSource {

    private static final ThreadLocal<String> contextHolder = new ThreadLocal<>();

    @Override
    protected Object determineCurrentLookupKey() {
        return contextHolder.get();
    }

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

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

    public static void clearDataSource() {
        contextHolder.remove();
    }
}
  • determineCurrentLookupKey() 方法返回當(dāng)前應(yīng)該使用的數(shù)據(jù)源標(biāo)識(shí)符。
  • 使用 ThreadLocal 變量存儲(chǔ)每個(gè)線程的數(shù)據(jù)源標(biāo)識(shí)符。

3.注冊(cè)動(dòng)態(tài)數(shù)據(jù)源

@Configuration
public class DynamicDataSourceConfig {

    @Autowired
    @Qualifier("primaryDataSource")
    private DataSource primaryDataSource;

    @Autowired
    @Qualifier("secondaryDataSource")
    private DataSource secondaryDataSource;

    @Bean(name = "dynamicDataSource")
    public DataSource dynamicDataSource() {
        Map<Object, Object> targetDataSources = new HashMap<>();
        targetDataSources.put("primary", primaryDataSource);
        targetDataSources.put("secondary", secondaryDataSource);

        DynamicDataSource dynamicDataSource = new DynamicDataSource();
        dynamicDataSource.setTargetDataSources(targetDataSources);
        dynamicDataSource.setDefaultTargetDataSource(primaryDataSource); // 設(shè)置默認(rèn)數(shù)據(jù)源
        return dynamicDataSource;
    }

    @Bean
    public SqlSessionFactory sqlSessionFactory(@Qualifier("dynamicDataSource") DataSource dataSource) throws Exception {
        SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();
        sessionFactory.setDataSource(dataSource);
        return sessionFactory.getObject();
    }
}

4.編寫攔截器

a.Spring MVC 攔截器

@Component
public class DataSourceInterceptor implements HandlerInterceptor {

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        // 根據(jù)請(qǐng)求路徑或其他條件設(shè)置數(shù)據(jù)源
        if (request.getRequestURI().startsWith("/api/secondary")) {
            DynamicDataSource.setDataSource("secondary");
        } else {
            DynamicDataSource.setDataSource("primary");
        }
        return true;
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        DynamicDataSource.clearDataSource(); // 清理線程局部變量
    }
}

b.MyBatis 攔截器

如果你想要基于 MyBatis 的 @Intercepts 注解來實(shí)現(xiàn),則可以這樣做:

@Intercepts({@Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class}),
             @Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})})
public class DataSourceInterceptor implements Interceptor {

    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        try {
            // 假設(shè)有一個(gè)方法可以根據(jù)某些條件確定要使用的數(shù)據(jù)源
            String dataSourceKey = determineDataSourceKey(invocation.getMethod().getName());
            DynamicDataSource.setDataSource(dataSourceKey);

            return invocation.proceed();
        } finally {
            DynamicDataSource.clearDataSource();
        }
    }

    @Override
    public Object plugin(Object target) {
        return Plugin.wrap(target, this);
    }

    @Override
    public void setProperties(Properties properties) {
    }
}

5.注冊(cè)攔截器

a.Spring MVC 攔截器注冊(cè)

@Configuration
public class WebConfig implements WebMvcConfigurer {

    @Autowired
    private DataSourceInterceptor dataSourceInterceptor;

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(dataSourceInterceptor).addPathPatterns("/**");
    }
}

b.MyBatis 攔截器注冊(cè)

如果你使用的是 MyBatis 攔截器,可以在配置類中添加如下代碼:

@Configuration
public class MyBatisConfig {

    @Autowired
    private DataSourceInterceptor myBatisDataSourceInterceptor;

    @Bean
    public SqlSessionFactory sqlSessionFactory(@Qualifier("dynamicDataSource") DataSource dataSource) throws Exception {
        SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();
        sessionFactory.setDataSource(dataSource);
        sessionFactory.setPlugins(new Interceptor[]{myBatisDataSourceInterceptor});
        return sessionFactory.getObject();
    }
}

?? 三、注意事項(xiàng)

  • 線程安全:由于我們使用了 ThreadLocal 來存儲(chǔ)數(shù)據(jù)源標(biāo)識(shí)符,因此在每次請(qǐng)求結(jié)束時(shí)都必須清理該變量,以防止內(nèi)存泄漏或跨請(qǐng)求污染。
  • 事務(wù)管理:如果涉及到事務(wù),需確保事務(wù)與數(shù)據(jù)源綁定正確。通常情況下,Spring 的事務(wù)管理器會(huì)自動(dòng)處理這個(gè)問題,但如果你手動(dòng)管理事務(wù),可能需要額外注意。
  • 性能考慮:頻繁地切換數(shù)據(jù)源可能會(huì)帶來一定的性能開銷,尤其是在高并發(fā)場(chǎng)景下。因此,應(yīng)盡量減少不必要的切換頻率。
  • 異常處理:確保在發(fā)生異常時(shí)也能正確地清理上下文,避免潛在的問題。

到此這篇關(guān)于Spring通過攔截器實(shí)現(xiàn)多數(shù)據(jù)源切換的示例代碼的文章就介紹到這了,更多相關(guān)Spring 攔截器多數(shù)據(jù)源切換內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論