使用Spring解决ibatis多数据源的苦恼

开发 后端
本文描述了Spring+iBatis多数据源的配置方案。在仅使用ibatis时,每多一个数据源就需要多一份sql-map-config配置文件。采用spring的AbstractRoutingDataSource就可以简单的解决这个问题。

iBatis多数据源的苦恼

在仅使用ibatis时,多数据源简直就是梦魇,每多一个数据源就需要多一份sql-map-config配置文件。

采用spring的AbstractRoutingDataSource就可以简单的解决这个问题。

AbstractRoutingDataSource实现了javax.sql.DataSource接口,因此可以理解为一个虚拟的动态DataSource,在需要的时候根据上下文Context动态决定使用哪个数据源。

Spring+iBatis多数据源的配置

下面是一个完整的配置:

< beans xmlns="http://www.springframework.org/schema/beans" 
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  xmlns:aop="http://www.springframework.org/schema/aop" 
  xmlns:tx="http://www.springframework.org/schema/tx" 
  xmlns:jee="http://www.springframework.org/schema/jee" 
  xsi:schemaLocation="  
   http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd  
   http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd  
   http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd  
   http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.5.xsd"> 
 
 
 < !-- ========================= GENERAL DEFINITIONS ========================= --> 
 < jee:jndi-lookup id="ds0" jndi-name="jdbc/ds0"/> 
 < jee:jndi-lookup id="ds1" jndi-name="jdbc/ds1"/> 
 < jee:jndi-lookup id="ds2" jndi-name="jdbc/ds2"/> 
   
 < bean id="dataSource" class="com.xxx.xxx.util.DynamicDataSource">    
         < property name="targetDataSources">    
            < map key-type="java.lang.String">    
               < entry key="0" value-ref="ds0"/> 
               < entry key="1" value-ref="ds1"/> 
               < entry key="2" value-ref="ds2"/>    
            < /map>    
         < /property>    
         < property name="defaultTargetDataSource" ref="1"/>    
 < /bean> 
        < !-- SqlMap setup for iBATIS Database Layer --> 
 < bean id="sqlMapClient" class="org.springframework.orm.ibatis.SqlMapClientFactoryBean"> 
  < property name="dataSource" ref="dataSource"/> 
  < property name="configLocation" value="classpath:com/xxx/xxx/dao/sqlmap/sql-map-config.xml"/> 
 < /bean> 
        < bean id="testDAO" class="com.xxx.xxx.dao.impl.TestDAO"> 
  < property name="sqlMapClient" ref="sqlMapClient"/> 
 < /bean> 
        < bean id="testService" class="com.xxx.xxx.service.impl.TestService"> 
  < property name="testDAO" ref="testDAO"/> 
 < /bean> 
< /beans> 
   
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.

其核心是DynamicDataSource,代码如下

package com.xxx.xxx.util;  
 
import org.apache.log4j.Logger;  
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;  
 
public class DynamicDataSource extends AbstractRoutingDataSource {  
 
 static Logger log = Logger.getLogger("DynamicDataSource");  
 @Override 
 protected Object determineCurrentLookupKey() {  
  // TODO Auto-generated method stub  
  return DbContextHolder.getDbType();  
 }  
 
}  
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.

上下文DbContextHolder为一线程安全的ThreadLocal,如下

package com.xxx.xxx.util;  
 
public class DbContextHolder {  
 private static final ThreadLocal contextHolder = new ThreadLocal();  
 
 public static void setDbType(String dbType) {  
  contextHolder.set(dbType);  
 }  
 
 public static String getDbType() {  
  return (String) contextHolder.get();  
 }  
 
 public static void clearDbType() {  
  contextHolder.remove();  
 }  
}  
   
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.

sql-map-config.xml如下

< ?xml version="1.0" encoding="UTF-8" standalone="no"?> 
< !DOCTYPE sqlMapConfig PUBLIC "-//iBATIS.com//DTD SQL Map Config 2.0//EN"  
        "http://www.ibatis.com/dtd/sql-map-config-2.dtd"> 
 
< sqlMapConfig> 
 
    < sqlMap resource="com/xxx/xxx/dao/sqlmap/Object.xml"/> 
 
< /sqlMapConfig> 
   
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.

这样在调用service之前只需要设置一下上下文即可调用相应的数据源,如下:

DbContextHolder.setDbType("2");  
//execute services  
//.........  
  • 1.
  • 2.
  • 3.

dao如下

package com.xxx.xxx.dao.impl;  
 
import java.util.HashMap;  
import java.util.List;  
import java.util.Map;  
 
import org.apache.log4j.Logger;  
import org.springframework.orm.ibatis.support.SqlMapClientDaoSupport;  
 
import com.xxx.xxx.vo.TestObj;  
 
public class TestDAO extends SqlMapClientDaoSupport implements ITestDAO {  
 
    static Logger log = Logger.getLogger(TestDAO.class);  
 
    public TestObj getTestObj(String objID) throws Exception {  
        return (TestObj) getSqlMapClientTemplate().queryForObject("getTestObj", objID);  
    }  
}  
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.

以上就介绍了Spring+iBatis多数据源的配置方案。本文来自eddysheng的博客:《基于spring和ibatis的多数据源切换方案》。

【编辑推荐】

  1. 操作更简单 iBatis配置文件方法
  2. 详细:iBatis配置实例
  3. iBATIS ParameterMap配置实例浅析
  4. iBATIS的多对多映射配置浅析
  5. iBATIS中添加DAO的配置浅析
责任编辑:yangsai 来源: eddysheng的博客
相关推荐

2020-12-31 07:55:33

spring bootMybatis数据库

2020-11-24 09:56:12

数据源读写分离

2022-05-18 12:04:19

Mybatis数据源Spring

2024-10-30 10:22:17

2023-09-07 08:39:39

copy属性数据源

2023-10-31 07:52:53

多数据源管理后端

2023-10-18 15:25:29

数据源数据库

2023-06-07 08:08:37

MybatisSpringBoot

2012-06-17 13:04:45

2020-03-13 14:05:14

SpringBoot+数据源Java

2020-06-02 07:55:31

SpringBoot多数据源

2023-01-04 09:33:31

SpringBootMybatis

2022-05-10 10:43:35

数据源动态切换Spring

2022-06-02 10:38:42

微服务数据源分布式

2022-12-19 07:21:35

Hutool-db数据库JDBC

2017-07-21 14:50:15

数据库DB分库事务处理

2023-01-10 16:30:22

Spring数据库

2025-01-17 09:11:51

2017-09-04 14:52:51

Tomcat线程数据源

2014-11-20 09:47:06

Java
点赞
收藏

51CTO技术栈公众号