无法找到Spring NamespaceHandler,用于XML架构命名空间[http://www.springframework.org/schema/batch]。

12

情况

我正在使用Spring Batch构建我们的数据仓库的累积快照,并且遇到了一个配置障碍,我无法解决。

我使用STS(SpringSource Tool Suite 2.8.1)创建了一个简单的Spring Batch项目,使用了Spring模板项目。这是我创建的两个xml配置文件:

launch-context.xml

<?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:batch="http://www.springframework.org/schema/batch"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:jdbc="http://www.springframework.org/schema/jdbc"
xsi:schemaLocation="
    http://www.springframework.org/schema/batch http://www.springframework.org/schema/batch/spring-batch-2.1.xsd
    http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-3.0.xsd
    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
    http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd">

<context:property-placeholder location="classpath:batch.properties" />

<context:component-scan base-package="edu.kdc.visioncards" />

<jdbc:initialize-database data-source="dataSource">
    <jdbc:script location="${batch.schema.script}" />
</jdbc:initialize-database>

<batch:job-repository id="jobRepository" />

<import resource="classpath:/META-INF/spring/module-context.xml" />

还有 module-context.xml

 <?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:batch="http://www.springframework.org/schema/batch"
xsi:schemaLocation="http://www.springframework.org/schema/batch http://www.springframework.org/schema/batch/spring-batch-2.1.xsd
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">

<description>Example job to get you started. It provides a skeleton for a typical batch application.</description>

<batch:job id="job1">
    <batch:step id="step1"  >           
        <batch:tasklet transaction-manager="transactionManager" start-limit="100" >
            <batch:chunk reader="reader" writer="writer" commit-interval="1" />
        </batch:tasklet>
    </batch:step>
</batch:job> 

</beans>

问题

我运行以下命令以编译和打包项目:mvn clean compile install

然后按照以下步骤进行:

  1. cdtarget文件夹

  2. 通过CommandLineJobRunner运行作业:java -jar batchprimer-1.0.jar META-INF/spring/module-context.xml job1

但我遇到以下错误:

C:\stsworkspace\BatchPrimer\target>java -jar batchprimer-1.0.jar META-INF/spring
/module-context.xml job1
2011-12-15 12:03:53,421 INFO [org.springframework.context.support.ClassPathXmlAp
plicationContext] - <Refreshing org.springframework.context.support.ClassPathXml
ApplicationContext@84abc9: startup date [Thu Dec 15 12:03:53 EST 2011]; root of
context hierarchy>
2011-12-15 12:03:53,468 INFO [org.springframework.beans.factory.xml.XmlBeanDefin
itionReader] - <Loading XML bean definitions from class path resource [META-INF/
spring/module-context.xml]>
2011-12-15 12:03:53,796 ERROR [org.springframework.batch.core.launch.support.Com
mandLineJobRunner] - <Job Terminated in error: Configuration problem: Unable to
locate Spring NamespaceHandler for XML schema namespace [http://www.springframew
ork.org/schema/batch]
Offending resource: class path resource [META-INF/spring/module-context.xml]
>
org.springframework.beans.factory.parsing.BeanDefinitionParsingException: Config
uration problem: Unable to locate Spring NamespaceHandler for XML schema namespa
ce [http://www.springframework.org/schema/batch]
Offending resource: class path resource [META-INF/spring/module-context.xml]

        at org.springframework.beans.factory.parsing.FailFastProblemReporter.err
or(FailFastProblemReporter.java:68)
        at org.springframework.beans.factory.parsing.ReaderContext.error(ReaderC
ontext.java:85)
        at org.springframework.beans.factory.parsing.ReaderContext.error(ReaderC
ontext.java:80)
        at org.springframework.beans.factory.xml.BeanDefinitionParserDelegate.er
ror(BeanDefinitionParserDelegate.java:284)
        at org.springframework.beans.factory.xml.BeanDefinitionParserDelegate.pa
rseCustomElement(BeanDefinitionParserDelegate.java:1335)
        at org.springframework.beans.factory.xml.BeanDefinitionParserDelegate.pa
rseCustomElement(BeanDefinitionParserDelegate.java:1328)
        at org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentRe
ader.parseBeanDefinitions(DefaultBeanDefinitionDocumentReader.java:135)
        at org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentRe
ader.registerBeanDefinitions(DefaultBeanDefinitionDocumentReader.java:93)
        at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.registe
rBeanDefinitions(XmlBeanDefinitionReader.java:493)
        at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.doLoadB
eanDefinitions(XmlBeanDefinitionReader.java:390)
        at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBea
nDefinitions(XmlBeanDefinitionReader.java:334)
        at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBea
nDefinitions(XmlBeanDefinitionReader.java:302)
        at org.springframework.beans.factory.support.AbstractBeanDefinitionReade
r.loadBeanDefinitions(AbstractBeanDefinitionReader.java:143)
        at org.springframework.beans.factory.support.AbstractBeanDefinitionReade
r.loadBeanDefinitions(AbstractBeanDefinitionReader.java:178)
        at org.springframework.beans.factory.support.AbstractBeanDefinitionReade
r.loadBeanDefinitions(AbstractBeanDefinitionReader.java:149)
        at org.springframework.beans.factory.support.AbstractBeanDefinitionReade
r.loadBeanDefinitions(AbstractBeanDefinitionReader.java:212)
        at org.springframework.context.support.AbstractXmlApplicationContext.loa
dBeanDefinitions(AbstractXmlApplicationContext.java:126)
        at org.springframework.context.support.AbstractXmlApplicationContext.loa
dBeanDefinitions(AbstractXmlApplicationContext.java:92)
        at org.springframework.context.support.AbstractRefreshableApplicationCon
text.refreshBeanFactory(AbstractRefreshableApplicationContext.java:130)
        at org.springframework.context.support.AbstractApplicationContext.obtain
FreshBeanFactory(AbstractApplicationContext.java:467)
        at org.springframework.context.support.AbstractApplicationContext.refres
h(AbstractApplicationContext.java:397)
        at org.springframework.context.support.ClassPathXmlApplicationContext.<i
nit>(ClassPathXmlApplicationContext.java:139)
        at org.springframework.context.support.ClassPathXmlApplicationContext.<i
nit>(ClassPathXmlApplicationContext.java:83)
        at org.springframework.batch.core.launch.support.CommandLineJobRunner.st
art(CommandLineJobRunner.java:282)
        at org.springframework.batch.core.launch.support.CommandLineJobRunner.ma
in(CommandLineJobRunner.java:574)

其他内容

在使用“mvn install”命令生成的目标文件夹中,我有以下内容:

  1. 一个classes文件夹
  2. 一个lib文件夹,其中包含所有maven依赖项
  3. 一个maven-archiver文件夹
  4. 一个test-classes文件夹
  5. 一个batchprimer-1.0.jar

在jar文件中,META-INF/MANIFEST.MF文件中有以下内容:

Manifest-Version: 1.0

Archiver-Version: Plexus Archiver

Created-By: Apache Maven

Built-By: dpardo

Build-Jdk: 1.6.0_20

Main-Class: org.springframework.batch.core.launch.support.CommandLineJ
 obRunner

Class-Path: lib/spring-jdbc-3.0.6.RELEASE.jar lib/spring-beans-3.0.6.R
 ELEASE.jar lib/spring-core-3.0.6.RELEASE.jar lib/spring-asm-3.0.6.REL
 EASE.jar lib/commons-logging-1.1.1.jar lib/spring-tx-3.0.6.RELEASE.ja
 r lib/aopalliance-1.0.jar lib/spring-aop-3.0.6.RELEASE.jar lib/spring
 -context-3.0.6.RELEASE.jar lib/spring-expression-3.0.6.RELEASE.jar li
 b/cglib-nodep-2.2.jar lib/spring-batch-core-2.1.7.RELEASE.jar lib/spr
 ing-batch-infrastructure-2.1.7.RELEASE.jar lib/xstream-1.3.jar lib/xp
 p3_min-1.1.4c.jar lib/jettison-1.1.jar lib/commons-io-1.4.jar lib/com
 mons-dbcp-1.2.2.jar lib/commons-pool-1.3.jar lib/hsqldb-1.8.0.7.jar l
 ib/aspectjrt-1.6.8.jar lib/aspectjweaver-1.6.8.jar lib/log4j-1.2.14.j
 ar lib/slf4j-log4j12-1.5.8.jar lib/slf4j-api-1.5.8.jar lib/mysql-conn
 ector-java-5.1.3.jar

因为在控制台中看起来它似乎找到了这些jar包,所以我不认为这是一个类路径的问题。

我尝试过的

基于错误信息,我粘贴了以下URL:

http://www.springframework.org/schema/batch/

Spring Batch的命名空间地址似乎是正确的。我没有在外面找到任何有用的东西。我在某个地方读到"某件事"可能会覆盖文件的命名空间,但我真的不理解为什么、什么和/或如何解决它。

这只是开箱即用,没有做任何修改、添加或其他操作就已经无法正常工作了。有什么想法吗?

谢谢

解决方法

除了下面标记为正确的解决方法之外,我还必须将launch-context.xml从组件扫描更改为常规的Spring连接方式,如下所示:

<!-- Commented this 
<context:property-placeholder location="classpath:batch.properties" />  


<jdbc:initialize-database data-source="dataSource">
    <jdbc:script location="${batch.schema.script}" />
</jdbc:initialize-database>
-->
<!-- Replace it with this below -->
<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}" />
</bean>

<bean id="transactionManager"
    class="org.springframework.jdbc.datasource.DataSourceTransactionManager"
    lazy-init="true">
    <property name="dataSource" ref="dataSource" />
</bean>

<bean id="placeholderProperties"
    class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
    <property name="location" value="classpath:batch.properties" />
    <property name="systemPropertiesModeName" value="SYSTEM_PROPERTIES_MODE_OVERRIDE" />
    <property name="ignoreUnresolvablePlaceholders" value="true" />
    <property name="order" value="1" />
</bean>

我认为Java只定位 META-INF/MANIFEST.MF (而不是 META-INF/spring/MANIFEST.MF) - 也许Java正在使用另一个清单文件。请查看此处了解如何自定义您的清单文件。我还发现 这篇文章,它有些相关,但您使用最新的Spring版本并且这种组合对我有效。 - dma_k
MANIFEST.MF 一直位于 META-INF 下面,抱歉我误导了。我会很快更新帖子。 - Viriato
通常我会放弃。如果你没有什么特别的,就把你的jar+libs上传到文件共享中。我会尝试在DefaultNamespaceHandlerResolver#getHandlerMappings()方法上设置断点并在调试器下运行它(顺便说一句,尝试将日志级别提高到DEBUG并查看此类报告的内容;应该看起来像这样)。 - dma_k
5个回答

28

通过CommandLineJobRunner运行作业:java -jar batchprimer-1.0.jar META-INF/spring/module-context.xml job1

即使有完整的目标文件夹,您仍需要为Java命令提供类路径信息,为了简化配置,您可以尝试使用一个包含所有必要库的全部一体化可执行jar文件,例如使用maven-shade-plugin或包含所有所需库的可执行shell脚本(.bat/.sh),例如使用appassembler-maven-plugin

maven-shade-plugin示例配置(创建额外的jar):

<plugin>
    <!-- create an all-in-one executable jar with maven-shade-plugin
         bound to phase:package 
         special handling for spring.handlers/spring.schemas files
         to prevent overwriting (maven-shade-plugin joins them to
         one file) 

         usage:
         cd to <project>/target
         java -jar hello-world-java-1.0-SNAPSHOT-executable.jar spring/batch/job/hello-world-job.xml helloWorldJob
         -->                     
    <artifactId>maven-shade-plugin</artifactId>
    <version>1.5</version>
    <executions>
        <execution>
            <phase>package</phase>
            <goals>
                <goal>shade</goal>
            </goals>
            <configuration>
                <transformers>
                    <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
                        <mainClass>org.springframework.batch.core.launch.support.CommandLineJobRunner</mainClass>
                    </transformer>
                    <transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
                        <resource>META-INF/spring.handlers</resource>
                    </transformer>
                    <transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
                        <resource>META-INF/spring.schemas</resource>
                    </transformer>
                </transformers>
                <shadedArtifactAttached>true</shadedArtifactAttached>
                <!-- configures the suffix name for the executable jar
                     here it will be '<project.artifact>-<project.version>-executable.jar'-->
                <shadedClassifierName>executable</shadedClassifierName>
            </configuration>
        </execution>
    </executions>
</plugin>

示例appassembler配置(创建子文件夹结构和.bat/.sh):

<plugin>
    <artifactId>appassembler-maven-plugin</artifactId>
    <groupId>org.codehaus.mojo</groupId>
    <version>1.1.1</version>
    <configuration>
        <repositoryLayout>flat</repositoryLayout>
        <installArtifacts>false</installArtifacts>
        <target>${project.build.directory}/appassembler</target>
        <defaultJvmSettings>
            <initialMemorySize>512M</initialMemorySize>
            <maxMemorySize>1024M</maxMemorySize>
            <extraArguments>
                <extraArgument>-Dlog4j.configuration=../etc/log4j/log4j.properties</extraArgument>
            </extraArguments>
        </defaultJvmSettings>
        <configurationDirectory>etc</configurationDirectory>
        <daemons>
            <daemon>
                <id>applicationName</id>
                <mainClass>org.springframework.batch.core.launch.support.CommandLineJobRunner</mainClass>
                <commandLineArguments>
                    <commandLineArgument>spring/job-runner.xml</commandLineArgument>
                    <commandLineArgument>helloWorldJob</commandLineArgument>
                    <commandLineArgument>input.file.pattern=file:.../**/*.txt</commandLineArgument>
                </commandLineArguments>
                <platforms>
                    <platform>booter-unix</platform>
                    <platform>booter-windows</platform>
                </platforms>
            </daemon>
        </daemons>
    </configuration>
    <executions>
        <execution>
            <phase>package</phase>
            <goals>
                <goal>generate-daemons</goal>
                <goal>create-repository</goal>
            </goals>
        </execution>
    </executions>
</plugin>

我已经在我的pom中添加了上述两个xml,并运行了以下命令:java -jar batchprimer-1.0.jar META-INF/spring/module-context.xml job1,但是我收到了从batchprimer-1.0.jar加载Main-Class清单属性失败的错误信息。有什么想法吗? - Viriato
哦,好的,batchprimer-1.0.jar里面没有清单。我不知道为什么这个shade插件会生成两个jar包。我在另一个它生成的jar包上运行了相同的命令 java -jar batchprimer-1.0-executable.jar META-INF/spring/module-context.xml job1,但是它抛出了一个Spring错误。我会排查一下,很奇怪,因为这是一个简单的项目,我什么都没动过。 - Viriato
shade插件会创建一个额外的可执行jar包,我会稍微修改一下描述,但这也在插件页面上说明了。 - Michael Pralow
1
+1 个解决方法不错,但我不想将所有的 jar 包合并成一个用于我的 Web 应用程序,而且 Spring 仍然可以解析批处理命名空间。 - dma_k
感谢所有为解决我的问题做出贡献的人。我通过使用第一个选项maven-shade-plugin解决了它,我的最终命令是java-jar batchprimer-1.0-executable.jar launch-context.xml job1。也许我一开始在命令行中甚至把xml写错了。 - Viriato

1
我使用的是Spring Batch 3.X.X版本和其他Spring依赖项4.X.X版本,并收到了上述错误。经过一些尝试和错误,我发现保持所有Spring版本相似可以解决我的问题。因此建议您使用Spring Batch 3.X.X版本或将Spring JDBC和Spring TX的依赖版本更改为2.X.X。
致敬,Rave

0
您需要在运行时类路径上添加spring-batch.jar。它提供了一个命名空间处理程序,可以处理该特定命名空间。

我有 spring-batch-core-2.1.7.RELEASE.jarspring-batch-infrastructure-2.1.7.RELEASE.jar 和其他的 spring-XXX-3.0.6.RELEASE.jar(s) 以及其他依赖项。但我仍然遇到了那个错误。 - Viriato
我在填满了类和一个名为lib的文件夹中运行了java -jar命令。该文件夹中包含所有的jar文件。清单文件有一个class-path条目,其中包含值:lib/spring-batch-core-2.1.7等。我将把它添加到我的帖子中。 - Viriato

0
在我的情况下,问题很简单,就是缺少了Spring事务依赖项,或者可能指向了旧的版本。将其替换为3.1版本后,错误消失了。

谢谢你的建议 - 对我来说,也是一个缺失的依赖项,而不是一个奇怪的设置问题。我忘记了spring-security-config。 - user1016765

0

出现错误的原因是spring-batch-core-2.1.7.RELEASE.jar中的META-INF/spring.handlers文件未被识别。该文件包含

http\://www.springframework.org/schema/batch=org.springframework.batch.core.configuration.xml.CoreNamespaceHandler

看起来可能是类路径问题,或者其他JAR(例如spring-context)中的spring.handlers文件在某种程度上占据优先地位。


谢谢,我如何控制顺序? - Viriato
@EmersonFarrugia:优先级不应该起到任何作用:所有资源都被扫描,并注册了命名空间处理程序。 - dma_k
@dma_k,你知道我能做些什么吗? - Viriato
@Viriato:请查看我的回复,首先确保您的根目录下的 MANIFEST.MF 文件是正确的。 - dma_k

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接