在实际的开发中遇到了需要2个数据源的情况。这里我们就需要配置多数据源了。
1 首先配置多数据源有2种实现方式
1)简单方式:一般情况下,我们可以配置多个数据源,然后为每个数据源写一套对应的sessionFactory和dao层代码我们称之为静态数据源配置
2)动态方式: 配置多个数据源,只对应一套sessionFactory,数据源之间可以动态切换。</span>
2 這两种方式相比较来看,第一种如果以后要配置多个数据源的话,对应都要配置sessionfactory. 第2种只需配置多个数据源自由切换对应一个sessionfactory 明显比第一个更装B一点。下面就是详细介绍我自己配置的第2中方法。首先是先配置2个数据源
<pre name="code" class="html"><beans xmlns="http://www.springframework.org/schema/beans"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-3.0.xsd ">
<description>公共的默认数据源配置</description>
<!-- portal master数据库 -->
<bean id="dataSource_master" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver" />
<property name="url" value="${db.master.dbUrl}" />
<property name="username" value="${db.master.userName}" />
<property name="password" value="${db.master.passWord}" />
<!-- 连接池启动时创建的初始化连接数量(默认值为0) -->
<property name="initialSize" value="1"/>
<!-- 连接池中可同时连接的最大的连接数 默认 8-->
<property name="maxActive" value="100"/>
<!-- 连接池中最小的空闲的连接数,低于这个数量会被创建新的连接 -->
<property name="minIdle" value="5"/>
<!-- 连接池中最大的空闲的连接数,超过的空闲连接将被释放,如果设置为负数表示不限制 默认8 -->
<property name="maxIdle" value="30"/>
<!-- 最大等待时间,当没有可用连接时,连接池等待连接释放的最大时间,超过该时间限制会抛出异常,如果设置-1表示无限等待 -->
<property name="maxWait" value="60000"/>
<!-- 超过removeAbandonedTimeout时间后,是否进 行没用连接(废弃)的回收(默认为false,调整为true)-->
<property name="removeAbandoned" value="true"/>
<!-- 超过时间限制,回收没有用(废弃)的连接(默认为 300秒,调整为180) -->
<property name="removeAbandonedTimeout" value="180"/>
<!-- 默认提交 -->
<property name="defaultAutoCommit" value="true"/>
<!-- 打开检查,用异步线程evict进行检查 -->
<property name="testWhileIdle" value="true"/>
<!-- 就是在进行borrowObject进行处理时,对拿到的connection进行validateObject校验 -->
<property name="testOnBorrow" value="false"/>
<!-- 就是在进行returnObject对返回的connection进行validateObject校验,个人觉得对数据库连接池的管理意义不大 -->
<property name="testOnReturn" value="false"/>
<!-- 代表检查的sql -->
<property name="validationQuery" value="select 1 from dual"/>
<!-- 代表在执行检查时,通过statement设置,statement.setQueryTimeout(validationQueryTimeout) -->
<property name="validationQueryTimeout" value="1"/>
<property name="timeBetweenEvictionRunsMillis" value="30000"/>
<!-- 代表每次检查链接的数量,建议设置和maxActive一样大,这样每次可以有效检查所有的链接. -->
<property name="numTestsPerEvictionRun" value="100"/>
</bean>
<!-- portal master数据库 -->
<bean id="dataSource_website" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver" />
<property name="url" value="${db.website.dbUrl}" />
<property name="username" value="${db.website.userName}" />
<property name="password" value="${db.website.passWord}" />
<!-- 连接池启动时创建的初始化连接数量(默认值为0) -->
<property name="initialSize" value="1"/>
<!-- 连接池中可同时连接的最大的连接数 默认 8-->
<property name="maxActive" value="100"/>
<!-- 连接池中最小的空闲的连接数,低于这个数量会被创建新的连接 -->
<property name="minIdle" value="5"/>
<!-- 连接池中最大的空闲的连接数,超过的空闲连接将被释放,如果设置为负数表示不限制 默认8 -->
<property name="maxIdle" value="30"/>
<!-- 最大等待时间,当没有可用连接时,连接池等待连接释放的最大时间,超过该时间限制会抛出异常,如果设置-1表示无限等待 -->
<property name="maxWait" value="60000"/>
<!-- 超过removeAbandonedTimeout时间后,是否进 行没用连接(废弃)的回收(默认为false,调整为true)-->
<property name="removeAbandoned" value="true"/>
<!-- 超过时间限制,回收没有用(废弃)的连接(默认为 300秒,调整为180) -->
<property name="removeAbandonedTimeout" value="180"/>
<!-- 默认提交 -->
<property name="defaultAutoCommit" value="true"/>
<!-- 打开检查,用异步线程evict进行检查 -->
<property name="testWhileIdle" value="true"/>
<!-- 就是在进行borrowObject进行处理时,对拿到的connection进行validateObject校验 -->
<property name="testOnBorrow" value="false"/>
<!-- 就是在进行returnObject对返回的connection进行validateObject校验,个人觉得对数据库连接池的管理意义不大 -->
<property name="testOnReturn" value="false"/>
<!-- 代表检查的sql -->
<property name="validationQuery" value="select 1 from dual"/>
<!-- 代表在执行检查时,通过statement设置,statement.setQueryTimeout(validationQueryTimeout) -->
<property name="validationQueryTimeout" value="1"/>
<property name="timeBetweenEvictionRunsMillis" value="30000"/>
<!-- 代表每次检查链接的数量,建议设置和maxActive一样大,这样每次可以有效检查所有的链接. -->
<property name="numTestsPerEvictionRun" value="100"/>
</bean>
4 然后是配置转换器
<bean id="data_dataSource" class="com.shangde.common.datesource.DynamicDataSource">
<property name="targetDataSources">
<map key-type="java.lang.String">
<entry key="dataSource_website" value-ref="dataSource_website" />
<entry key="dataSource_master" value-ref="dataSource_master" />
</map>
</property>
<property name="defaultTargetDataSource" ref="dataSource_master" />
</bean>
5配置切面
<!-- 动态数据源切换aop 先于事务的aop -->
<bean id="dataSourceInterceptor" class="com.shangde.common.datesource.DataSourceInterceptor" />
<aop:config>
<aop:aspect id="dataSourceAspect" ref="dataSourceInterceptor">
<aop:pointcut id="dsMaster" expression="execution(* com.shangde.master.mobile.*.service..*.*(..))" />
<aop:pointcut id="dsWebsite" expression="execution(* com.shangde.master.website.*.service..*.*(..))" />
<aop:before method="setdataSourceMaster" pointcut-ref="dsMaster"/>
<aop:before method="setdataSourceWebsite" pointcut-ref="dsWebsite"/>
</aop:aspect>
</aop:config>
6.配置sessionfactory
<bean id="sqlSessionFactory_master" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="data_dataSource" />
<!-- 全局sqlMapConfig.xml公共配置文件路径 -->
<property name="configLocation" value="classpath:/mybatis_master.xml" />
<property name="typeAliasesPackage" value="com.shangde.master.dmo.*" />
</bean>
7
AbstractRoutingDataSource转换器代码
DynamicDataSource
package com.shangde.common.datesource;
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
public class DynamicDataSource extends AbstractRoutingDataSource {
@Override
protected Object determineCurrentLookupKey() {
return CustomerContextHolder.getCustomerType();
}
}
DataSourceInterceptor
package com.shangde.common.datesource;
import org.aspectj.lang.JoinPoint;
public class DataSourceInterceptor {
public void setdataSourceMaster(JoinPoint jp) {
CustomerContextHolder.setCustomerType("dataSource_master");
}
public void setdataSourceWebsite(JoinPoint jp) {
CustomerContextHolder.setCustomerType("dataSource_website");
}
}
CustomerContextHolder
package com.shangde.common.datesource;
public class CustomerContextHolder {
public static final String DATA_SOURCE_MASTER = "dataSource_master";
public static final String DATA_SOURCE_WEBSITE = "dataSource_website";
private static final ThreadLocal<String> contextHolder = new ThreadLocal<String>();
public static void setCustomerType(String customerType) {
contextHolder.set(customerType);
}
public static String getCustomerType() {
return contextHolder.get();
}
public static void clearCustomerType() {
contextHolder.remove();
}
}
8 好了代码就这么多了 一些失误的配置自行解决。
原理: AbstractRoutingDataSource 能够通过key的值在 map中寻找对应的 值,也就是对应的数据源。然而怎么才能获得key的值呢?。我们对不同数据源分包(package)管理,同一包下的代码使用了同一数据源。通过AOP来实现,大家都知道web开发常见是三层结构:controller、service、dao。假设我们创建了2个包 package1 ,package2分别对应不同的数据源db1 和db2。在对应的包下面创建 我们 service 包 和 dao 包 ,我们只需要用AOP切 不同数据源下的包名的service的目录即可,当controller层调用service 类时,发现它在 package1 .service 包下,这是spring 切面就会调用 切面类,设置代表数据源的字符串db1然后AbstractRoutingDataSource根据db1這个key在map中找到对应的数据源 db1 .然后我们只是找到了這个数据源,那么怎么用呢。
我们发现AbstractRoutingDataSource extends AbstractDataSource 发现他继承自 AbstractDataSource。而且刚才我们已经找到了我们到底要那个数据源,这样我们就可以直接使用 這个实现了 AbstractRoutingDataSource 的类作为数据源了。