问题:
使用mybatis时都是用的sqlmapper来做的数据库到java对象的映射,因此在针对一些特定数据库方言使用时无法在多个数据库上切换。
解决方案:
思路:
通过定义environment的id来指定使用不同的数据库映射文件,如下
<!--WizRtf2Html Charset=0 -->
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
<environments
default="mysql">
<environment id="mysql">
<transactionManager
type="JDBC"
/>
<dataSource type="POOLED">
<property name="driver" value="com.mysql.jdbc.Driver"
/>
<property name="url" value="jdbc:mysql://localhost:3306/test"
/>
<property name="username" value="root"
/>
<property name="password" value="pwd"
/>
</dataSource>
</environment>
</environments>
<mappers>
<mapper
resource="cn/dcr/mybatis/entity/UserMapper.xml"
/>
</mappers>
</configuration>
environment的id是mysql映射文件是cn/dcr/mybatis/entity/UserMapper.xml
那么mybatis实际是读取的mysql/cn/dcr/mybatis/entity/UserMapper.xml
实现:
以org.apache.ibatis.builder.xml.XMLConfigBuilder类为蓝本创建一个新类org.apache.ibatis.builder.xml.XMLConfigBuilderEx
添加一个成员变量
private String dialect;
修改environmentsElement方法设置方言
private void
environmentsElement(XNode context) throws
Exception {
if (context != null) {
if (environment == null)
{
environment =
context.getStringAttribute("default");
}
for (XNode child :
context.getChildren()) {
String id = child.getStringAttribute("id");
dialect = id.toLowerCase();//设置方言
if (isSpecifiedEnvironment(id))
{
TransactionFactory txFactory = transactionManagerElement(child.evalNode("transactionManager"));
DataSourceFactory
dsFactory = dataSourceElement(child.evalNode("dataSource"));
Environment.Builder
environmentBuilder = new
Environment.Builder(id).transactionFactory(txFactory).dataSource(dsFactory.getDataSource());
configuration.setEnvironment(environmentBuilder.build());
}
}
}
}
修改mapperElement方法
private void
mapperElement(XNode parent) throws Exception
{
if (parent != null) {
for (XNode child :
parent.getChildren()) {
String resource = child.getStringAttribute("resource");
String url = child.getStringAttribute("url");
InputStream
inputStream;
if (resource
!= null && url
== null) {
if(dialect !=
null){
resource = dialect
+ "/" + resource;//从方言指定位置查找
}
ErrorContext.instance().resource(resource);
inputStream
=
Resources.getResourceAsStream(resource);
XMLMapperBuilder
mapperParser = new
XMLMapperBuilder(inputStream, configuration, resource,
configuration.getSqlFragments());
mapperParser.parse();
}
else if (url
!= null &&
resource == null) {
if(dialect !=
null){
url = dialect + "/" + url;//从方言指定位置查找
}
ErrorContext.instance().resource(url);
inputStream
=
Resources.getUrlAsStream(url);
XMLMapperBuilder
mapperParser = new
XMLMapperBuilder(inputStream, configuration, url,
configuration.getSqlFragments());
mapperParser.parse();
}
else {
throw new
BuilderException("A mapper element may only specify a url or
resource, but not
both.");
}
}
}
}
继承org.apache.ibatis.session.SqlSessionFactoryBuilder类创建一个新类org.apache.ibatis.session.SqlSessionFactoryBuilderEx用来加载org.apache.ibatis.builder.xml.XMLConfigBuilderEx
覆盖父类中的build方法
public SqlSessionFactory build(InputStream
inputStream, String environment, Properties props) {
try {
XMLConfigBuilderEx parser = new
XMLConfigBuilderEx(inputStream, environment,
props);
Configuration config =
parser.parse();
return
build(config);
} catch (Exception
e) {
throw
ExceptionFactory.wrapException("Error building
SqlSession.", e);
} finally
{
ErrorContext.instance().reset();
try
{
inputStream.close();
} catch (IOException e) {
// Intentionally ignore. Prefer previous
error.
}
}
}
调用org.apache.ibatis.builder.xml.XMLConfigBuilderEx来加载配置文件
思路:
自定义mybatis配置加载Bean在spring的配置文件中添加方言的配置让自定义Bean在加载mybatis的配置时可以使用不同的数据库映射文件,如下
<bean
id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBeanEx">
<property
name="dataSource"
ref="dataSource" />
<property name="configLocation"
value="classpath:configuration.xml" />
<property
name="mapperLocations" value="classpath*:${jdbc.dialect}/mappers/*.xml"
/>
</bean>
${jdbc.dialect}在配置文件中设置,如mysql,那么spring会把mysql/mappers下的所有映射文件加载进来
实现:
以org.mybatis.spring.SqlSessionFactoryBean为蓝本创建org.mybatis.spring.SqlSessionFactoryBeanEx类
将成员变量
private SqlSessionFactoryBuilder
sqlSessionFactoryBuilder = new
SqlSessionFactoryBuilder();
修改为
private SqlSessionFactoryBuilderEx
sqlSessionFactoryBuilderEx = new
SqlSessionFactoryBuilderEx();
并去掉setSqlSessionFactoryBuilder方法添加setSqlSessionFactoryBuilderEx方法
覆盖buildSqlSessionFactory方法
protected SqlSessionFactory buildSqlSessionFactory()
throws IOException, IllegalAccessException,
InstantiationException {
XMLConfigBuilderEx
xmlConfigBuilderEx;
Configuration
configuration;
if (this.configLocation !=
null) {
try
{
xmlConfigBuilderEx = new XMLConfigBuilderEx(this.configLocation.getInputStream(), null, this.configurationProperties);
configuration
= xmlConfigBuilderEx.parse();
} catch (Exception ex) {
throw new
NestedIOException("Failed to parse config resource: "
+ this.configLocation, ex);
} finally
{
ErrorContext.instance().reset();
}
if (this.logger.isDebugEnabled())
{
this.logger.debug("Parsed configuration file: '" +
this.configLocation + "'");
}
} else {
if (this.logger.isDebugEnabled())
{
this.logger.debug("Property 'configLocation' not specified, using default MyBatis
Configuration");
}
configuration = new
Configuration();
}
if (this.transactionFactory == null) {
this.transactionFactory =
new SpringManagedTransactionFactory(this.dataSource);
}
Environment
environment = new
Environment(this.environment, this.transactionFactory, this.dataSource);
configuration.setEnvironment(environment);
if (!ObjectUtils.isEmpty(this.mapperLocations)) {
Map<String, XNode>
sqlFragments = new
HashMap<String, XNode>();
for (Resource mapperLocation : this.mapperLocations)
{
if (mapperLocation == null) {
continue;
}
// MyBatis holds a Map using "resource" name as a
key.
// If a mapper file is
loaded, it searches for a mapper
// interface type.
// If the type is found then it tries to load the mapper
file
// again looking for
this:
//
// String
xmlResource = type.getName().replace('.', '/') +
// ".xml";
//
// So if a mapper
interface exists, resource cannot be an
// absolute path.
//
Otherwise MyBatis will throw an exception
because
// it will load both a
mapper interface and the mapper xml file,
// and throw an exception telling that a mapperStatement
cannot
// be loaded
twice.
String path;
if (mapperLocation instanceof ClassPathResource)
{
path = ((ClassPathResource)
mapperLocation).getPath();
} else {
//
this won't work if there is also a mapper interface
in
//
classpath
path =
mapperLocation.toString();
}
try {
XMLMapperBuilder
xmlMapperBuilder = new
XMLMapperBuilder(mapperLocation.getInputStream(), configuration, path,
sqlFragments);
xmlMapperBuilder.parse();
}
catch (Exception e)
{
throw new NestedIOException("Failed to
parse mapping resource: '" + mapperLocation
+ "'",
e);
} finally
{
ErrorContext.instance().reset();
}
if (this.logger.isDebugEnabled())
{
this.logger.debug("Parsed mapper
file: '" + mapperLocation + "'");
}
}
}
else {
if (this.logger.isDebugEnabled())
{
this.logger.debug("Property 'mapperLocations' was not specified, only MyBatis mapper
files specified in the config xml were
loaded");
}
}
return this.sqlSessionFactoryBuilderEx.build(configuration);
}
分享到:
相关推荐
此外,如果你的项目使用了Spring框架,那么可以利用Spring的多数据源支持与Mybatis结合,通过Spring的 `AbstractRoutingDataSource` 类来实现动态数据源切换。`AbstractRoutingDataSource` 可以根据某种策略(如 ...
MyBatis-Plus 支持的数据库包括: * MySQL * MariaDB * Oracle * DB2 * H2 * HSQL * SQLite * Postgre * SQLServer * Presto * Gauss * Firebird * Phoenix * ClickHouse * Sybase ASE * OceanBase * 达梦数据库 * ...
"spring+hibernate和spring+myBatis实现连接多个数据库,同时操作的项目"是针对这种需求的一个解决方案,旨在提供一种灵活且动态的数据源切换机制。 首先,Spring框架作为Java领域中最受欢迎的应用框架之一,其强大...
完整的Demo结合了springmvc——mybatis,实现了工具类文件上传下载,结合了Redis的初步使用,并且能使用threadlocal实现数据库动态切换,很适合初建项目做参考,适合初学者使用。
在IT行业中,数据库连接管理是系统性能优化的关键环节之一,MyBatis作为一款优秀的持久层框架,配合Druid这样的高效数据库连接池,可以提供高效、稳定的数据库操作支持。本话题聚焦于"Mybatis+Druid Association ...
在SpringBoot项目中,整合Mybatis-Plus并实现多数据源的动态切换,同时支持分页查询是一项常见的需求。以下将详细阐述这个过程中的关键步骤和技术要点。 首先,我们需要引入必要的Maven依赖。这里提到了四个关键...
在现代企业级应用开发...总的来说,Spring Boot结合JPA或MyBatis实现多数据源动态切换,不仅提高了系统的灵活性,还便于进行数据库扩展和管理。理解和掌握这一技术,对于提升系统设计能力和解决复杂问题具有重要意义。
在现代企业级应用开发中,数据源管理是一个关键部分,特别是在多租户或者需要根据业务需求动态切换数据库的场景下。本教程将详细介绍如何在Spring Boot项目中整合Druid数据源池与Mybatis,实现多数据源切换的功能,...
主数据库用于处理写操作,从数据库用于处理读操作,这样可以将读压力分散到多个从库,提高系统整体性能。通常,主库的数据会实时同步到从库,确保数据一致性。主从复制的实现方式包括异步复制和半同步复制等,具体...
springboot实现数据源动态切换 注意事项: 1. 该demo采用yml配置数据库信息,注意url标签为jdbc-url 2.项目中加了日志输出,可看到完整执行过程 3.在Service中应用事务时,自定义的注解将失效,解决办法:可将注解...
在企业级应用开发中,动态数据源是一种常见需求,它允许程序在运行时根据不同的业务逻辑切换到不同的数据库。在本项目中,我们将探讨如何利用MyBatis与Spring框架实现动态切换数据源的功能。首先,我们需要理解...
这个项目结合了四个关键的技术组件,它们分别是Mybatis、Spring、SpringMVC和Quartz,下面将详细介绍这些技术以及它们在多数据源切换和跨数据库同步中的应用。 **Mybatis** 是一款轻量级的Java持久层框架,它允许...
本项目"springboot+mybatis动态切换数据源完整项目架构"正是针对这种需求提供的一种解决方案,它支持MySQL和Oracle数据库,使得开发者可以轻松地在两者之间进行数据源的切换,以适应不同的业务场景。 首先,我们要...
Spring 和 MyBatis 结合使用时,实现多数据源动态切换是一项重要的技术。本文将深入探讨如何在 Spring 中配置和管理多个数据源,并实现动态切换。 首先,我们需要理解“多数据源”是什么。它是指在一个应用中同时...
综上所述,这个项目通过SpringBoot、Gradle和MyBatis,实现了一个支持多数据源动态切换的应用。开发者可以利用AOP在运行时根据业务逻辑选择合适的数据源,提高了系统的灵活性和可扩展性。同时,Gradle的使用使得依赖...
然而,标准的MyBatis注解并不直接支持切换不同的数据库。为了实现数据库切库,我们需要创建一个自定义注解,用于标记需要切换数据库的方法,并编写相应的拦截器来处理这些注解。 1. **创建自定义注解**: 创建一个...
以上就是在Spring Boot中利用注解动态切换多数据源的基本步骤。这种设计使得我们的代码更加模块化,可以根据业务需求灵活地选择和切换数据源,提高了代码的可维护性和可扩展性。在实际项目中,还可以结合事务管理、...
MybatisPlus是在Mybatis基础上扩展的轻量级ORM(对象关系映射)框架,它简化了SQL查询和实体对象的绑定,使得开发者可以更专注于业务逻辑,而不用过多地关注数据库操作的细节。在这个项目中,MybatisPlus用于执行...
这种技术在分布式系统、微服务架构中非常常见,可以方便地支持多个数据库的切换,如读写分离、数据库分片等场景。 3. **自定义数据源配置**: 自定义数据源配置通常涉及到Spring框架中的`DataSource`接口实现,如...
对于存在主从关系的多数据源场景,通常需要实现数据源的动态切换,以便根据具体的业务逻辑来决定使用哪个数据库执行查询或更新操作。这种情况下,可以利用Spring的AOP功能来实现数据源的动态选择。 - **创建数据...