将 JUnit JPA 测试从 Spring 2.5.5 迁移到 Spring 3.0.4

Posted

技术标签:

【中文标题】将 JUnit JPA 测试从 Spring 2.5.5 迁移到 Spring 3.0.4【英文标题】:Migrating JUnit JPA tests from Spring 2.5.5 to Spring 3.0.4 【发布时间】:2011-04-22 22:34:50 【问题描述】:

我们的项目刚刚决定从 Spring 2.5.5 迁移到 Spring 3.0.4。在我的原始代码中,我的单元测试如下所示:

public class RequisitionDaoTest extends AbstractJpaTests


    public static String FAIL_MSG_UNEXPECTED_ERROR = "FAIL: Unexpected Error";

        @PersistenceContext
        private EntityManager em;

    @PersistenceUnit
    private EntityManagerFactory emf;

        @Test
        public void testSomething ()
        
          ...
        

这在 Spring 2.5.5 中运行良好。现在在 Spring 3.0.4 中,AbstractJpaTests 类已被弃用。

问题是我似乎找不到用什么来替换它。我看过几篇显示使用名为 AbstractTransactionalJUnit4SpringContextTests 的帖子,但是当我转换为使用它时,出现以下错误:

严重:在允许 TestExecutionListener [org.springframework.test.context.support.DependencyInjectionTestExecutionListener@4aec4aec] 准备测试实例 [RequsitionDaoTest@5dcc5dcc] 时捕获异常
org.springframework.beans.factory.BeanCreationException:创建名为“RequsitionDaoTest”的bean时出错:注入自动装配的依赖项失败;嵌套异常是 org.springframework.beans.factory.BeanCreationException:无法自动装配方法:public void org.springframework.test.context.junit4.AbstractTransactionalJUnit4SpringContextTests.setDataSource(javax.sql.DataSource);嵌套异常是 org.springframework.beans.factory.NoSuchBeanDefinitionException:没有为依赖项找到类型为 [javax.sql.DataSource] 的匹配 bean:预计至少有 1 个 bean 有资格作为此依赖项的自动装配候选者。依赖注释:
    在 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessPropertyValues(AutowiredAnnotationBeanPostProcessor.java:286)
    在 org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.populateBean(AbstractAutowireCapableBeanFactory.java:1074)
    在 org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.autowireBeanProperties(AbstractAutowireCapableBeanFactory.java:374)
    在 org.springframework.test.context.support.DependencyInjectionTestExecutionListener.injectDependencies(DependencyInjectionTestExecutionListener.java:110)
    在 org.springframework.test.context.support.DependencyInjectionTestExecutionListener.prepareTestInstance(DependencyInjectionTestExecutionListener.java:75)
    在 org.springframework.test.context.TestContextManager.prepareTestInstance(TestContextManager.java:333)
    在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner.createTest(SpringJUnit4ClassRunner.java:220)
    在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner$1.runReflectiveCall(SpringJUnit4ClassRunner.java:301)
    在 org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
    在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner.methodBlock(SpringJUnit4ClassRunner.java:303)
    在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:240)
    在 org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:49)
    在 org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
    在 org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
    在 org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
    在 org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
    在 org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
    在 org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
    在 org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
    在 org.junit.runners.ParentRunner.run(ParentRunner.java:236)
    在 org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:180)
    在 org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:45)
    在 org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:460)
    在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:673)
    在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:386)
    在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:196)
原因:org.springframework.beans.factory.BeanCreationException:无法自动装配方法:public void org.springframework.test.context.junit4.AbstractTransactionalJUnit4SpringContextTests.setDataSource(javax.sql.DataSource);嵌套异常是 org.springframework.beans.factory.NoSuchBeanDefinitionException:没有为依赖项找到类型为 [javax.sql.DataSource] 的匹配 bean:预计至少有 1 个 bean 有资格作为此依赖项的自动装配候选者。依赖注释:
    在 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredMethodElement.inject(AutowiredAnnotationBeanPostProcessor.java:608)
    在 org.springframework.beans.factory.annotation.InjectionMetadata.inject(InjectionMetadata.java:84)
    在 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessPropertyValues(AutowiredAnnotationBeanPostProcessor.java:283)
    ... 26 更多
原因:org.springframework.beans.factory.NoSuchBeanDefinitionException:没有为依赖项找到类型为 [javax.sql.DataSource] 的匹配 bean:预计至少有 1 个 bean 有资格作为此依赖项的自动装配候选者。依赖注释:
    在 org.springframework.beans.factory.support.DefaultListableBeanFactory.raiseNoSuchBeanDefinitionException(DefaultListableBeanFactory.java:920)
    在 org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:789)
    在 org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:703)
    在 org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor$AutowiredMethodElement.inject(AutowiredAnnotationBeanPostProcessor.java:559)
    ... 28 更多

由于我使用的是 JPA,因此我没有或不需要 Spring 的 DataSource bean。

那么对于新 Spring 中的 AbstractJpaTests 类有没有直接的类比,还是我现在应该继续使用已弃用的类?

谢谢...

--史蒂夫

【问题讨论】:

使用不推荐使用的类并不是坏事,它只是意味着现在有另一种首选方式。 @skaffman 没错,但是当我们迁移时,我倾向于尝试删除已弃用的内容,因为我知道在某些时候它会被删除。 【参考方案1】:

其实,在新的 Spring 中并没有类似的 AbstractJpaTests 类。新的测试类不需要从 Spring 继承任何行为来完成他们的工作。现在全部由注释处理。

所以我的班级实际上应该是这样的:

公开课 RequisitionDaoTest


    公共静态字符串 FAIL_MSG_UNEXPECTED_ERROR = "失败:意外错误";

        @PersistenceContext
        私人 EntityManager em;

    @PersistenceUnit
    私人 EntityManagerFactory 电动势;

        @测试
        公共无效testSomething()
        
          ...
        

此外,您需要包含来自 Spring 的断言函数,因为它们不包含在 AbstractJpaTests 类中。导入看起来像这样:

导入静态 org.junit.Assert.*;

希望这可以帮助遇到类似问题的其他人。

【讨论】:

以上是关于将 JUnit JPA 测试从 Spring 2.5.5 迁移到 Spring 3.0.4的主要内容,如果未能解决你的问题,请参考以下文章

Spring JPA Junit 关闭自动回滚

使用 spring-data-jpa 和 MockMvc 进行 spring boot junit 测试

Spring Boot 5 多个 JUnit JPA 测试文件接线

Spring Boot JPA 元模型不能为空!尝试运行 JUnit / 集成测试时

在 Spring Boot 应用程序的 JUnit 测试中,自动装配的 JPA 存储库没有合格的 bean

防止在 JUnit 中执行 jpa @Query