Mybatis 多数据库支持实践(MySQL+Oracle)

根据这篇博客去尝试 https://blog.csdn.net/qq_35981283/article/details/79503571

但是发现始终读取不到 databaseId

经过调试找到了原因。

首先从 mybatis 的自动配置说起

@Configuration
@ConditionalOnClass({SqlSessionFactory.class, SqlSessionFactoryBean.class})
@ConditionalOnBean({DataSource.class})
@EnableConfigurationProperties({MybatisProperties.class})
@AutoConfigureAfter({DataSourceAutoConfiguration.class})
public class MybatisAutoConfiguration {

@ConditionalOnClass是Springboot实现自动配置的重要支撑之一。其用途是判断当前classpath下是否存在指定类,若是则将当前的配置装载入spring容器。

ConditionalOnBean则是查看是否有这个 bean,也就是这个配置加载的条件是有 SqlSessionFactory 这个类以及 DataSource 的 bean,这也就是为啥需要@AutoConfigureAfter 保证 DataSourceAutoConfiguration 这个类先加载了。

然后就是看 MybatisAutoConfiguration 的实现,里面有一个方法是初始化 sqlSessionFactory

@Bean
    @ConditionalOnMissingBean
    public SqlSessionFactory sqlSessionFactory(DataSource dataSource) throws Exception {
        SqlSessionFactoryBean factory = new SqlSessionFactoryBean();
        factory.setDataSource(dataSource);
        factory.setVfs(SpringBootVFS.class);
        if (StringUtils.hasText(this.properties.getConfigLocation())) {
            factory.setConfigLocation(this.resourceLoader.getResource(this.properties.getConfigLocation()));
        }

        factory.setConfiguration(this.properties.getConfiguration());
        if (!ObjectUtils.isEmpty(this.interceptors)) {
            factory.setPlugins(this.interceptors);
        }

        if (this.databaseIdProvider != null) {
            factory.setDatabaseIdProvider(this.databaseIdProvider);
        }

        if (StringUtils.hasLength(this.properties.getTypeAliasesPackage())) {
            factory.setTypeAliasesPackage(this.properties.getTypeAliasesPackage());
        }

        if (StringUtils.hasLength(this.properties.getTypeHandlersPackage())) {
            factory.setTypeHandlersPackage(this.properties.getTypeHandlersPackage());
        }

        if (!ObjectUtils.isEmpty(this.properties.resolveMapperLocations())) {
            factory.setMapperLocations(this.properties.resolveMapperLocations());
        }

        return factory.getObject();
    }

在这个方法里判断如果已经有 databaseIdProvider 这个 bean的话,会给这个 SQLSessionFactory 设置一个dataBaseIdProvider。而我恰恰在 spring boot 启动的时候加载了这么一个 bean

 @Bean
    public DatabaseIdProvider getDatabaseIdProvider(){
        DatabaseIdProvider databaseIdProvider = new VendorDatabaseIdProvider();
        Properties properties = new Properties();
        properties.setProperty("Oracle","oracle");
        properties.setProperty("MySQL","mysql");
        properties.setProperty("DB2","db2");
        properties.setProperty("Derby","derby");
        properties.setProperty("H2","h2");
        properties.setProperty("HSQL","hsql");
        properties.setProperty("Informix","informix");
        properties.setProperty("MS-SQL","ms-sql");
        properties.setProperty("PostgreSQL","postgresql");
        properties.setProperty("Sybase","sybase");
        properties.setProperty("Hana","hana");
        databaseIdProvider.setProperties(properties);
        return databaseIdProvider;
    }

一切看上去合情合理,那是为啥呢?打个断点在sqlSessionFactory方法中,发现果然没到这个方法中。那就说明这个bean 已经有了,搜索代码,果然代码的其他地方也加载了sqlSessionFactory,而那个方法里没有去设置 databasIdProvider,果断加入,问题解决。

猜你喜欢

转载自blog.csdn.net/Jacob_Zheng/article/details/88389242