使用Spring Batch Admin时,它会尝试为dataSource,transactionManager等提供一些默认值。
如果要覆盖这些默认值,可以在 META-INF / spring / batch / servlet / override / 文件夹下创建自己的xml bean定义,并在引导期间保证默认属性将被覆盖。
在spring-batch-admin中,dataSource缺省值在data-source-context.xml中定义,带有此定义
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="${batch.jdbc.driver}" />
<property name="url" value="${batch.jdbc.url}" />
<property name="username" value="${batch.jdbc.user}" />
<property name="password" value="${batch.jdbc.password}" />
<property name="testWhileIdle" value="${batch.jdbc.testWhileIdle}"/>
<property name="validationQuery" value="${batch.jdbc.validationQuery}"/>
</bean>
现在,我想用JNDI数据源覆盖此dataSource,因此我删除了batch.jdbc.driver
,batch.jdbc.url
等属性行,并具有以下jndi定义
<bean id="dataSource" class="org.springframework.jndi.JndiObjectFactoryBean">
<property name="jndiName" value="java:comp/env/jdbc/dbconn" />
</bean>
您可能很容易猜到系统首先尝试初始化data-source-context.xml中定义的dataSource bean,因为它无法找到属性值batch.jdbc。*的任何值,它会失败并出现异常。
无法在字符串值[$ {batch.jdbc.driver}]中解析占位符'batch.jdbc.driver'
由于我将使用JNDI并且不想处理这些属性值,所以我无法继续。
在这种情况下如何覆盖dataSource?
答案 0 :(得分:6)
在Spring Batch Admin中,有2个正在加载的Spring ApplicationContexts:
servlet-config.xml包含以下导入:
<import resource="classpath*:/META-INF/spring/batch/servlet/resources/*.xml" />
<import resource="classpath*:/META-INF/spring/batch/servlet/manager/*.xml" />
<import resource="classpath*:/META-INF/spring/batch/servlet/override/*.xml" />
webapp-config.xml包含以下导入:
<import resource="classpath*:/META-INF/spring/batch/bootstrap/**/*.xml" />
<import resource="classpath*:/META-INF/spring/batch/override/**/*.xml" />
servlet-config.xml配置servlet,webapp-config.xml配置应用程序的后端部分。 问题是dataSource bean是第二个配置中的/定义的一部分,而不是第一个配置。 因此,当您将dataSource bean添加到servlet配置的覆盖(/ META-INF / spring / batch / servlet / override / * .xml)时,就像您一样,将新bean添加到第一个上下文中,而不是覆盖第二个上下文的dataSource bean。
因此,您需要将自定义data-source-context.xml置于META-INF / spring / batch / override /而不是META-INF / spring / batch / servlet / override /
然后它可以工作,你甚至不会得到Could not resolve placeholder 'batch.jdbc.driver' in string value [${batch.jdbc.driver}]
错误。
答案 1 :(得分:3)
自Spring 3.1以来,'profiles'功能允许您根据您所处的环境设置数据源“源”(一个用于本地测试的嵌入式,一个用于部署的JNDI)。
这看起来像下面这样;
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:jee="http://www.springframework.org/schema/jee"
xmlns:jdbc="http://www.springframework.org/schema/jdbc"
xsi:schemaLocation="http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-3.2.xsd
http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-3.2.xsd
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<!-- "production" datasource -->
<jee:jndi-lookup id="dataSource" jndi-name="java:comp/env/jdbc/dbconn"/>
<!-- profile for "local" testing -->
<beans profile="local">
<!-- datasource that only gets created in that active profile -->
<jdbc:embedded-database id="dataSource" type="H2"/>
</beans>
</beans>
在这个例子中,当'active profile'设置为“local”时,它会覆盖jndi-lookup数据源。