Spring JDBC 多数据源管理

因为项目须要从已经运行的多个项目的数据库中取值,因此就出现了须要访问多个数据源的状况。 java

Spring配置文件 mysql

<!-- 属性文件读入 -->
	<bean id="propertyConfigurer"
		class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
		<property name="locations">
			<list>
				<value>classpath*:resources/spring/database.properties</value>
			</list>
		</property>
	</bean>

	<bean id="jdbcTemplate"
		class="org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate">
		<constructor-arg ref="dynamicDataSource" />
	</bean>

	<bean id="dynamicDataSource" class="com.primemis.datasource.DynamicDataSource">
		<!-- 经过key-value的形式来关联数据源 -->
		<property name="targetDataSources">
			<map key-type="com.primemis.datasource.DatabaseType">
				<entry key="DEFAULT" value-ref="defaultDataSource"></entry>
				<entry key="COUNTER" value-ref="counterDataSource"></entry>
				<entry key="XPOS" value-ref="xposDataSource"></entry>
				<entry key="CONXPOS" value-ref="conxposDataSource"></entry>
				<entry key="ESPOS" value-ref="esposDataSource"></entry>
			</map>
		</property>
		<property name="defaultTargetDataSource" ref="defaultDataSource">
		</property>
	</bean>


	<!--MySql 数据源配置 Bgn -->
	<bean id="defaultDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
		destroy-method="close">
		<property name="driverClass" value="${mysql.driverclass}"></property>
		<property name="jdbcUrl" value="${mysql.jdbcurl}"></property>
		<property name="user" value="${mysql.user}"></property>
		<property name="password" value="${mysql.password}"></property>
		<property name="checkoutTimeout" value="${jdbc.checkoutTimeout}"></property>
		<property name="acquireIncrement" value="${jdbc.acquireIncrement}"></property>  <!-- 当链接池中的链接用完时,C3P0一次性建立新链接的数目2 -->
		<property name="initialPoolSize" value="${jdbc.initialPoolSize}"></property>  <!-- 初始化时建立的链接数,必须在minPoolSize和maxPoolSize之间 -->
		<property name="minPoolSize" value="${jdbc.minPoolSize}"></property>
		<property name="maxPoolSize" value="${jdbc.maxPoolSize}"></property>
		<!-- 最大空闲时间,超过空闲时间的链接将被丢弃 [须要注意:mysql默认的链接时长为8小时(28800)【可在my.ini中添加 wait_timeout=30(单位秒)设置链接超时】,这里设置c3p0的超时必须<28800] -->
		<property name="maxIdleTime" value="${jdbc.maxIdleTime}"></property>
		<!-- <property name="idleConnectionTestPeriod" value="60"></property> --> <!-- 每60秒检查链接池中的空闲链接 -->
		<!-- <property name="maxStatements" value="20"></property> -->  <!-- jdbc的标准参数 用以控制数据源内加载的PreparedStatement数量,但因为预缓存的Statement属 于单个Connection而不是整个链接 -->
	</bean>
	<!--MySql 数据源配置 End -->

	<!--客流系统 Sql Server 数据源配置 Bgn -->
	<bean id="counterDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
		destroy-method="close">
		<property name="driverClass" value="${counter.driverclass}"></property>
		<property name="jdbcUrl" value="${counter.jdbcurl}"></property>
		<property name="user" value="${counter.user}"></property>
		<property name="password" value="${counter.password}"></property>
		<property name="checkoutTimeout" value="${jdbc.checkoutTimeout}"></property>
		<property name="acquireIncrement" value="${jdbc.acquireIncrement}"></property>  <!-- 当链接池中的链接用完时,C3P0一次性建立新链接的数目2 -->
		<property name="initialPoolSize" value="${jdbc.initialPoolSize}"></property>  <!-- 初始化时建立的链接数,必须在minPoolSize和maxPoolSize之间 -->
		<property name="minPoolSize" value="${jdbc.minPoolSize}"></property>
		<property name="maxPoolSize" value="${jdbc.maxPoolSize}"></property>
		<!-- 最大空闲时间,超过空闲时间的链接将被丢弃 [须要注意:mysql默认的链接时长为8小时(28800)【可在my.ini中添加 wait_timeout=30(单位秒)设置链接超时】,这里设置c3p0的超时必须<28800] -->
		<property name="maxIdleTime" value="${jdbc.maxIdleTime}"></property>
		<!-- <property name="idleConnectionTestPeriod" value="60"></property> --> <!-- 每60秒检查链接池中的空闲链接 -->
		<!-- <property name="maxStatements" value="20"></property> -->  <!-- jdbc的标准参数 用以控制数据源内加载的PreparedStatement数量,但因为预缓存的Statement属 于单个Connection而不是整个链接 -->
	</bean>

	<!--客流系统 Sql Server 数据源配置 End -->

	<!-- Sybase JDBC信息配置 Bgn -->
	<bean id="aDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
		<property name="driverClass" value="${Sybase.driverClassName}">
		</property>
		<property name="jdbcUrl" value="${a.url}"></property>
		<property name="user" value="${a.userName}"></property>
		<property name="password" value="${a.password}"></property>
	</bean>
	<bean id="bDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
		<property name="driverClass" value="${Sybase.driverClassName}">
		</property>
		<property name="jdbcUrl" value="${b.url}"></property>
		<property name="user" value="${b.userName}"></property>
		<property name="password" value="${b.password}"></property>
	</bean>
	<bean id="cDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
		<property name="driverClass" value="${Sybase.driverClassName}">
		</property>
		<property name="jdbcUrl" value="${c.url}"></property>
		<property name="user" value="${c.userName}"></property>
		<property name="password" value="${c.password}"></property>
	</bean>
	<!-- Sybase JDBC信息配置 End -->
DynamicDataSource类

import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;

public class DynamicDataSource extends AbstractRoutingDataSource {

	@Override
	protected Object determineCurrentLookupKey() {
		// TODO Auto-generated method stub
		return DatabaseContextHolder.getDatabaseType();
	}

}
DatabaseContextHolder类

import org.springframework.util.Assert;

public class DatabaseContextHolder {

	private static final ThreadLocal<DatabaseType> contextHolder = new ThreadLocal<DatabaseType>();

	public static void setDatabaseType(DatabaseType DatabaseType) {
		Assert.notNull(DatabaseType, "DatabaseType cannot be null");
		contextHolder.set(DatabaseType);
	}

	public static DatabaseType getDatabaseType() {
		return (DatabaseType) contextHolder.get();
	}

	public static void clearDatabaseType() {
		contextHolder.remove();
	}
}
Dao使用方式

@Transactional(propagation = Propagation.REQUIRED, readOnly = false, rollbackFor = Exception.class)
	public int update(String sql, SqlParameterSource paramSource,
			DatabaseType type) {
		DatabaseContextHolder.setDatabaseType(type);
		int rows = this.jdbc.update(sql, paramSource);
		DatabaseContextHolder.clearDatabaseType();
		return rows;
	}
相关文章
相关标签/搜索