使用Spring和Maven进行测试:applicationContext

时间:2012-04-11 10:41:25

标签: spring unit-testing maven

这个问题似乎是世界的问题,但我仍然无法找到解决方案......

我试图进行简单的测试:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"/applicationContext.xml", "/PersonsPopulateTest-context.xml"})
@Transactional
public class PersonsPopulateTest {

文件位于:

src
   main
      resources
           applicationContext.xml

src        
   test
      resources
          PersonsPopulateTest-context.xml 

因此,构建这些文件后,目标/类目标/测试类

但是mvn test命令仍然说:无法加载ApplicationContext

官方文件说:

@RunWith(SpringJUnit4ClassRunner.class)
// ApplicationContext will be loaded from "/applicationContext.xml" and "/applicationContext-test.xml"
// in the root of the classpath
@ContextConfiguration(locations={"/applicationContext.xml", "/applicationContext-test.xml"})
public class MyTest {
    // class body...
}

我哪里出错了?

谢谢, Vlaidimir

更新。万无一失的报告:

java.lang.IllegalStateException: Failed to load ApplicationContext
at org.springframework.test.context.TestContext.getApplicationContext(TestContext.java:157)
at org.springframework.test.context.support.DependencyInjectionTestExecutionListener.injectDependencies(DependencyInjectionTestExecutionListener.java:109)
at org.springframework.test.context.support.DependencyInjectionTestExecutionListener.prepareTestInstance(DependencyInjectionTestExecutionListener.java:75)
at org.springframework.test.context.TestContextManager.prepareTestInstance(TestContextManager.java:321)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.createTest(SpringJUnit4ClassRunner.java:211)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner$1.runReflectiveCall(SpringJUnit4ClassRunner.java:288)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.methodBlock(SpringJUnit4ClassRunner.java:290)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:231)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:71)
at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:174)
at org.apache.maven.surefire.junit4.JUnit4TestSet.execute(JUnit4TestSet.java:53)
at org.apache.maven.surefire.junit4.JUnit4Provider.executeTestSet(JUnit4Provider.java:123)
at org.apache.maven.surefire.junit4.JUnit4Provider.invoke(JUnit4Provider.java:104)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at org.apache.maven.surefire.util.ReflectionUtils.invokeMethodWithArray(ReflectionUtils.java:164)
at org.apache.maven.surefire.booter.ProviderFactory$ProviderProxy.invoke(ProviderFactory.java:110)
at org.apache.maven.surefire.booter.SurefireStarter.invokeProvider(SurefireStarter.java:175)
at org.apache.maven.surefire.booter.SurefireStarter.runSuitesInProcessWhenForked(SurefireStarter.java:107)
at org.apache.maven.surefire.booter.ForkedBooter.main(ForkedBooter.java:68)
Caused by: java.lang.IllegalArgumentException: Can not load an ApplicationContext with a NULL 'contextLoader'. Consider annotating your test class with @ContextConfiguration.
at org.springframework.util.Assert.notNull(Assert.java:112)
at org.springframework.test.context.TestContext.loadApplicationContext(TestContext.java:117)
at org.springframework.test.context.TestContext.getApplicationContext(TestContext.java:148)
... 30 more

10 个答案:

答案 0 :(得分:10)

我认为Maven根本没有包含main / resources中的XML文件。

您可以尝试明确指定要在pom.xml中包含的内容。

如果以下配置有效,请告诉我:

    <!-- Add this directly after the <build>-opening tag -->
    <resources>
        <resource>
            <filtering>true</filtering>
            <directory>src/test/resources</directory>
            <includes>
                <include>**/*.properties</include>
            </includes>
            <excludes>
                <exclude>**/*local.properties</exclude>
            </excludes>
        </resource>
        <resource>
            <directory>src/main/resources</directory>
            <includes>
                <include>**/*.properties</include>
                <include>**/*.xml</include>
            </includes>
        </resource>
    </resources>

这是我在你的情况下使用的东西。如果您没有要包含的属性文件,可以编辑它。

答案 1 :(得分:9)

我的测试上下文文件位于 src \ test \ resources \ spring 文件夹下。 我设法用

加载上下文
@ContextConfiguration(locations={"classpath:**/test-context.xml"})

但是引用(在test-context.xml中)到 src \ main \ resources \ spring 文件夹下的application-context.xml失败

我设法通过在测试类中使用

创建ClassPathXmlApplicationContext来加载应用程序上下文
ClassPathXmlApplicationContext appContext=new ClassPathXmlApplicationContext(new String[]{"classpath:spring/application-context.xml","classpath:spring/model-context.xml"});

如果这有助于或可能会产生任何其他问题,请告诉我。

答案 2 :(得分:3)

我认为最佳做法是将您的应用程序上下文文件用于测试src / test / resources下的PersonsPopulateTest-context.xml。此文件将被复制到目标/测试类中,您可以在测试类中引用它,如下所示:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:**/PersonsPopulateTest-context.xml"})
@Transactional
public class PersonsPopulateTest {

}

如果您仍想在src / main / resources下引用applicationContext.xml,则必须按如下方式包含它:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"file:src/main/resources/applicationContext.xml"})
@Transactional
public class PersonsPopulateTest {

}

它对我有用。

答案 3 :(得分:2)

<!-- Add this directly after the <build>-opening tag in the your pom-->
            <testResources>
                <testResource>
                    <directory>src/test/resources</directory>
                    <filtering>true</filtering>
                    <includes>
                        <include>**/*.xml</include>
                        <include>**/*.properties</include>
                    </includes>
                </testResource>
            </testResources>

答案 4 :(得分:2)

您的应用程序上下文必须包含在类路径中并放入*:

@ContextConfiguration(locations = { "classpath:*/application-context.xml" })

答案 5 :(得分:1)

出于某种原因,我遇到了同样的问题,当我使用JDK6而不是JDK8运行maven测试时(我的情况下这是一个遗留应用程序),它有效。

如果它对任何人有帮助。

答案 6 :(得分:1)

我遇到了同样的问题。对我来说,测试是通过eclipse成功运行的。但是,当我运行mvn test时,它给了我错误:无法加载ApplicationContext

  

修复:将src / test / resources目录添加到surefire的classpath中   插件为 -

<plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-surefire-plugin</artifactId>
            <version>2.10</version>
            <configuration>
                <additionalClasspathElements>
                    <additionalClasspathElement>${project.basedir}/src/test/resources</additionalClasspathElement>
                </additionalClasspathElements>      
            </configuration>

Adding classpath to SureFire

希望它有所帮助。

答案 7 :(得分:0)

我遇到了同样的问题,所有文件都成功复制到目标/类目标/测试类,仍然 spring 无法找到他们。

当我在 pom

中明确指定 maven-surefire-plugin maven-resources-plugin 的版本时,问题就消失了

答案 8 :(得分:0)

更新: 实际上在我的情况下,问题是在Compile on Save选项启用。我使用Netbeans,选项设置为For test execution only。此值编译已更改的文件,并用新文件替换资源。但是,由于除了测试资源之外还使用了应用程序资源,For test execution only会在target文件夹中生成错误生成的资源。

更改Compile on Save=For test execution only

Compile on Save=Disable 修复了问题。

下面的文字也是正确的,但有时不起作用。它只在我重新启动Netbeans IDE之前工作。但是,问题原因的细节是正确的,因此,我更愿意离开测试。


OLD: 在我的情况下,appContext-test.xml中有src/main/resources。 当我更改任何代码并启动一个单元测试(并非所有代码)时,它会重新编译并正确执行。但是,如果我再次启动相同的单元测试,则java.lang.IllegalStateException: Failed to load ApplicationContext会失败。

我已从

更改了pom.xml
<build>
    <resources>
        <resource>
            <directory>${project.basedir}/src/main/resources</directory>
            <filtering>true</filtering>
        </resource>
    </resources>
    <testResources>
        <testResource>
            <directory>${project.basedir}/src/test/java/resources</directory>
            <filtering>true</filtering>
        </testResource>
    </testResources>
</build>

<build>
    <resources>
        <resource>
            <directory>${project.basedir}/src/main/resources</directory>
            <filtering>true</filtering>
        </resource>
    </resources>
    <testResources>
        <testResource>
            <directory>${project.basedir}/src/main/resources</directory>
            <filtering>true</filtering>
        </testResource>
        <testResource>
            <directory>${project.basedir}/src/test/java/resources</directory>
            <filtering>true</filtering>
        </testResource>
    </testResources>
</build>

现在一切正常。

{strong>错误到期到appContext-test.xml使用src/main/resources/my.properties文件包含很多变量,例如

database.url = ${database.url}
database.username = ${database.username}
database.password = ${database.password}

在构建期间填充。但是,如果您在src/main/resources中跳过testResource,则my.properties会按原样添加到target/classes/my.properties,例如。没有替代。这个文件当然打破了上下文。

PS:你可以删除${project.basedir}/ - 这是我的习惯。

答案 9 :(得分:0)

在您的集成测试类中,尝试添加

@Test
public void contextLoads(){
}