gpt4 book ai didi

java - 动态改变 Spring 数据源

转载 作者:IT老高 更新时间:2023-10-28 13:46:05 24 4
gpt4 key购买 nike

我有一个 Spring 应用程序,我想动态更改数据源,即。当输入 DS URL 时,Spring bean 和所有依赖项将自动更新。我知道这有点奇怪,但无论如何我想实现这一点。我的 Spring 配置如下:

<bean id="majorDataSource" class="org.postgresql.ds.PGSimpleDataSource">
<property name="serverName" value="${jdbc.serverName}" />
<property name="portNumber" value="${jdbc.portNumber}" />
<property name="user" value="${jdbc.username}" />
<property name="password" value="${jdbc.password}" />
<property name="databaseName" value="${jdbc.databaseName}" />
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="majorDataSource"/>
</bean>

<tx:annotation-driven transaction-manager="transactionManager"/>

<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="majorDataSource"/>
<property name="configLocation" value="classpath:sqlmap-config.xml"/>
</bean>

问题是:

  1. JDBC URL 存储在属性中,可以在运行时更改。

  2. 一旦 URL 被更改,我需要重新创建数据源,可能还需要重新创建依赖对象。我不知道如何在 Spring 中优雅地做到这一点?

我知道Spring确实可以基于一键动态路由数据源,但是数据源URL是在Spring中预定义的,不会改变运行时。这不是我的情况。

最佳答案

你可以使用spring的AbstractRoutingDataSource通过扩展它并覆盖方法determineCurrentLookupKey(),该方法应该返回引用数据源的spring bean的键。

看看这个blog article on spring source's blog这将向您展示如何使用该功能的示例。

基本上要回答您的问题,您需要做的是在您的 XML 配置中将两个数据源定义为不同的 spring bean。无需动态创建,spring 会同时加载两者,并根据 determineCurrentLookupKey() 方法中的条件动态使用其中一个。

这会导致类似:

XML 配置

<!-- first data source -->
<bean id="majorDataSource" class="org.postgresql.ds.PGSimpleDataSource">
<property name="serverName" value="${jdbc.major.serverName}" />
<property name="portNumber" value="${jdbc.major.portNumber}" />
<property name="user" value="${jdbc.major.username}" />
<property name="password" value="${jdbc.major.password}" />
<property name="databaseName" value="${jdbc.major.databaseName}" />
</bean>
<!-- second data source -->
<bean id="minorDataSource" class="org.postgresql.ds.PGSimpleDataSource">
<property name="serverName" value="${jdbc.minor.serverName}" />
<property name="portNumber" value="${jdbc.minor.portNumber}" />
<property name="user" value="${jdbc.minor.username}" />
<property name="password" value="${jdbc.minor.password}" />
<property name="databaseName" value="${jdbc.minor.databaseName}" />
</bean>
<!-- facade data source -->
<bean id="dataSource" class="blog.datasource.CustomerRoutingDataSource">
<property name="targetDataSources">
<map>
<entry key="MINOR" value-ref="minorDataSource"/>
<entry key="MAJOR" value-ref="majorDataSource"/>
</map>
</property>
<property name="defaultTargetDataSource" ref="majorDataSource"/>
</bean>
<!-- wiring up -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<tx:annotation-driven transaction-manager="transactionManager"/>
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<property name="configLocation" value="classpath:sqlmap-config.xml"/>
</bean>

Java

public class MyRoutingDataSource extends AbstractRoutingDataSource {
@Override
protected Object determineCurrentLookupKey() {
// get the current url
HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest();
if (request.getRequestURL().toString().endsWith("/minor"))
return "MINOR";
else
return "MAJOR";
}
}

关于java - 动态改变 Spring 数据源,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13507522/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com