使用spring配置多个数据源

转载 2015年07月06日 22:23:42

1.数据源定义

        这里以2个c3p0数据库连接池的数据源作为实例。在spring框架下需要加入c3p0的依赖。这里以数据同步为例:

        (1)数据库来源的连接池数据源配置

<bean id="dataSourceFrom" class="com.mchange.v2.c3p0.ComboPooledDataSource">  
    <property name="driverClass" value="${jdbc.driver}" />  
    <property name="jdbcUrl" value="${jdbc.from.url}" />  
    <property name="user" value="${jdbc.from.username}" />  
    <property name="password" value="${jdbc.from.password}" />  
    <property name="autoCommitOnClose" value="true" />  
    <property name="checkoutTimeout" value="${cpool.checkoutTimeout}" />  
    <property name="initialPoolSize" value="${cpool.minPoolSize}" />  
    <property name="minPoolSize" value="${cpool.minPoolSize}" />  
    <property name="maxPoolSize" value="${cpool.maxPoolSize}" />  
    <property name="maxIdleTime" value="${cpool.maxIdleTime}" />  
    <property name="acquireIncrement" value="${cpool.acquireIncrement}" />  
    <property name="maxIdleTimeExcessConnections" value="${cpool.maxIdleTimeExcessConnections}" />  
</bean>  
        (2)数据插入库的连接池数据源配置

<bean id="dataSourceTo" class="com.mchange.v2.c3p0.ComboPooledDataSource">  
    <property name="driverClass" value="${jdbc.driver}" />  
    <property name="jdbcUrl" value="${jdbc.from.url}" />  
    <property name="user" value="${jdbc.from.username}" />  
    <property name="password" value="${jdbc.from.password}" />  
    <property name="autoCommitOnClose" value="true" />  
    <property name="checkoutTimeout" value="${cpool.checkoutTimeout}" />  
    <property name="initialPoolSize" value="${cpool.minPoolSize}" />  
    <property name="minPoolSize" value="${cpool.minPoolSize}" />  
    <property name="maxPoolSize" value="${cpool.maxPoolSize}" />  
    <property name="maxIdleTime" value="${cpool.maxIdleTime}" />  
    <property name="acquireIncrement" value="${cpool.acquireIncrement}" />  
    <property name="maxIdleTimeExcessConnections" value="${cpool.maxIdleTimeExcessConnections}" />  
</bean>  

2.扩展Spring的AbstractRoutingDataSource

        AbstractRoutingDataSource中的抽象方法determineCurrentLookupKey是实现数据源的route的核心,这里对该方法进行重写

public class DynamicDataSource extends AbstractRoutingDataSource{  
  
    @Override  
    protected Object determineCurrentLookupKey() {  
        return DBContextHolder.getDBType();  
    }  
}  
       其中DBContextHolder是一个线程安全的ThreadLocal,具体代码如下:

public class DBContextHolder{  
    public static final String DATA_SOURCE_FROM = "dataSourceFrom";  
    public static final String DATA_SOURCE_TO = "dataSourceTo";  
      
    private static final ThreadLocal contextHolder = new ThreadLocal();  
      
    public static void setDBType(String dbType) {  
        contextHolder.set(dbType);  
    }  
      
    public static String getDBType() {  
        return contextHolder.get();  
    }  
      
    public static void clearDBType() {  
        contextHolder.remove();  
    }  
}

3.配置动态数据源

        将DynamicDataSource加入到spring的配置文件中,同时配置DynamicDataSource的targetDataSource(也就是多个数据源)属性的Map映射

<bean id="dynamicDataSource" class="datasource.DynamicDataSource" >  
      
    <property name="targetDataSources">  
        <map>  
            <entry value-ref="dataSourceFrom" key="dataSourceFrom"></entry>  
            <entry value-ref="dataSourceTo" key="dataSourceTo"></entry>  
        </map>  
    </property>  
    <property name="defaultTargetDataSource" ref="dataSourceFrom" />  
</bean>   

4.使用动态数据源

        说明:例子中的DynamicDataSource继承自AbstractRoutingDataSource,而AbstractRoutingDataSource有继承自AbstractDataSource,AbstractDataSource实现了统一的DataSource接口,所以DynamicDataSource同样可以当做一个DataSource来使用

        这里使用ibatis作为持久层框架

   <bean id="sqlMapClient" class="org.springframework.orm.ibatis.SqlMapClientFactoryBean">  
        <property name="dataSource" ref="dynamicDataSource<span style="font-family: Arial, Helvetica, sans-serif;">"</span><span style="font-family: Arial, Helvetica, sans-serif;">/>  </span>
        <property name="configLocation" value="classpath:com/xxx/xxx/dao/sqlmap/sql-map-config.xml"/>  
    </bean>  
        <bean id="userDAO" class="com.xxx.xxx.dao.impl.UserDAO">  
        <property name="sqlMapClient" ref="sqlMapClient"/>  
    </bean> 

5.事务管理

        使用动态数据源的时候,事务管理和单数据源没有差别

    <bean id="txManager"  
        class="org.springframework.jdbc.datasource.DataSourceTransactionManager">  
        <property name="dataSource" ref="dynamicDataSource" <span style="font-family: Arial, Helvetica, sans-serif;">/>  </span>
    </bean>  
  
    <tx:advice id="txAdvice" transaction-manager="txManager">  
        <tx:attributes>  
            <tx:method name="save*" propagation="REQUIRED" />  
            <tx:method name="delete*" propagation="REQUIRED" />  
            <tx:method name="update*" propagation="REQUIRED" />  
            <tx:method name="select*" propagation="REQUIRED" />  
            <tx:method name="*" propagation="REQUIRES_NEW"  
                rollback-for="Exception" />  
        </tx:attributes>  
    </tx:advice>  
  
    <aop:config proxy-target-class="false">  
        <aop:advisor pointcut="execution(* com.mycompany.app.service.impl.*.*(..))"  
            advice-ref="txAdvice" />  
    </aop:config>  
  
    <bean id="sqlMapClient" class="org.springframework.orm.ibatis.SqlMapClientFactoryBean">  
        <property name="configLocation">  <!-- name 为configLocation或s 不可为其他 -->  
            <value>sqlMap.xml</value> <!-- 不区分大小写,路径前可加'/' -->  
        </property>  
        <!-- dataSource不是必需 -->  
        <property name="dataSource">  
            <ref local="dynamicDataSource"<span style="font-family: Arial, Helvetica, sans-serif;"> />  </span>
        </property>  
    </bean> 


6.使用

        (1)手动控制

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");  
BaseDAO dao = (BaseDAO) context.getBean("sqlBaseDAO", BaseDAOImpl.class);  
  
try {  
    DBContextHolder.setCustomerType(DBContextHolder.DATA_SOURCE_FROM);  
    System.err.println(dao.select("select count(*) sum from TEST t ").get(0).get("SUM"));  
    DBContextHolder.setCustomerType(DBContextHolder.DATA_SOURCE_TO);  
    System.err.println(dao.select("select count(*) sum from TEST t ").get(0).get("SUM"));  
      
} catch (Exception e) {  
    e.printStackTrace();  
}  

        (2)使用AOP控制
@Aspect  
public class DynamicDataSourceAspect {  
    @Pointcut("execution (public service.impl..*.*(..))")  
    public void serviceExecution(){}  
      
    @Before("serviceExecution()")  
    public void setDynamicDataSource(JoinPoint jp) {  
        for(Object o : jp.getArgs()) {  
            //处理具体的逻辑 ,根据具体的境况CustomerContextHolder.setCustomerType()选取DataSource  
        }  
    }  
}  


7.注意

        必须在开启事务之前就切换好数据源,因为事务是和数据源绑定的。

        如事务控制设置在service层,那么切换数据源就必须在上一层,也就是Controller层


Spring配置多个数据源

  • 2011年10月19日 16:36
  • 8KB
  • 下载

多数据源配置-使用spring配置多个数据源实现读写分离

我们在很多的项目中经常会有用到多个数据源。比如数据库读写分离,读操作都去从库里读,写操作都往主库里写。那么这里主库和从库就是两个不同的数据源。再比如要做两个数据库之间的数据转换,从一个数据库读取数据写...

spring 配置多个数据源(基于JPA)

http://www.springframework.org/schema/beans"     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instan...
  • al_jin
  • al_jin
  • 2015年12月07日 10:04
  • 4098

spring boot 配置MyBatis,支持多个数据源和分页插件

spring boot中的MyBatis配置是比较复杂的。 下面总结针对mySql数据库的配置和使用的详细过程(有两个数据库:名字为test和my_db):...

spring+ibatis 配置多个数据源,jdbc 和 jndi 实现

需求:一个项目中需要链接多个数据源,(操作多个数据库数据表) 项目实战:项目是采用spring + ibatis,采用jetty容器实现。 针对jdbc和jndi实现:          jdbc我是...
  • lsblsb
  • lsblsb
  • 2014年01月08日 17:29
  • 1971

Spring3+Hibernate3(Jpa) 配置多个数据源的解决方案(基于注解)

----------------配置文件-------------------------------- -------------------------------------META-IN...

Spring配置一个事务中多个数据源

(个人观点)   在一个事务中包含多个数据源的更新操作时,需要JTA的支持,通常JTA的支持是由J2EE容器提供的(WAS,Weblogic...),也有单独实现JTA的第3方jar。     ...

Spring3+Jpa 配置多个数据源的解决方案(基于注解)

补充说明:不同的应用场景,解决方案也不同。例如下文的配置实例,是创建了两个事务管理器。如果项目要求必须只能存在一个事务管理,解决方案可参考:http://today.java.net/pub/a/to...

Spring3+Hibernate3(Jpa) 配置多个数据源的解决方案(基于注解)

----------------配置文件-------------------------------- -------------------------------------META-INF/...

spring mvc + mybatis配置多个数据源问题。

惯例: 我是温浩然: 用过spring MVC的都知道,项目的每一小块,都分成,controller,dao,service,module,基本是这四块,分别是,实现需求的控制器,数据库操作的方法,实...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:使用spring配置多个数据源
举报原因:
原因补充:

(最多只允许输入30个字)