将ehcache添加到DAO类后Junit测试失败,无法在测试类中实例化DAO

Posted

技术标签:

【中文标题】将ehcache添加到DAO类后Junit测试失败,无法在测试类中实例化DAO【英文标题】:Junit test failing after adding ehcache to a DAO class, can't instantiate DAO in test class 【发布时间】:2017-08-23 13:24:44 【问题描述】:

我正在将 ehcache 添加到我的项目中,如果我在我的 DAO 上注释 ehcache 注释,我的 DAO 单元测试类运行正常,如果我取消注释它们会失败,说明它无法自动装配 DAO bean .

这是我得到的错误:

org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'com.nicobrest.kamehouse.dao.DragonBallUserDaoJpaTest': Injection of autowired dependencies failed;
nested exception is org.springframework.beans.factory.BeanCreationException: Could not autowire field: private com.nicobrest.kamehouse.dao.DragonBallUserDaoJpa com.nicobrest.kamehouse.dao.DragonBallUserDaoJpaTest.dragonBallUserDaoJpa;
nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type [com.nicobrest.kamehouse.dao.DragonBallUserDaoJpa] found for dependency:
expected at least 1 bean which qualifies as autowire candidate for this dependency. Dependency annotations: @org.springframework.beans.factory.annotation.Autowired(required=true), @org.springframework.beans.factory.annotation.Qualifier(value=dragonBallUserDaoJpa)
    at org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor.postProcessPropertyValues(AutowiredAnnotationBeanPostProcessor.java:334)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.populateBean(AbstractAutowireCapableBeanFactory.java:1214)
    at
...

我也尝试过,而不是在单元测试中自动装配 DAO bean,而是自动装配应用程序上下文并按名称获取 bean,当我这样做时,我得到一个异常,它不能将 $proxy32 投射到我的DAO 类。

java.lang.ClassCastException: com.sun.proxy.$Proxy32 cannot be cast to com.nicobrest.kamehouse.dao.DragonBallUserDaoJpa
    at com.nicobrest.kamehouse.dao.DragonBallUserDaoJpaTest.setUp(DragonBallUserDaoJpaTest.java:71)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
    at java.lang.reflect.Method.invoke(Unknown Source)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:24)
    at 
...

这些是相关文件:

applicationContext.xml

...
<cache:annotation-driven cache-manager="cacheManager"/>
<import resource="applicationContext-persistence.xml" />

<!-- Ehcache configuration -->
<bean id="cacheManager" class="org.springframework.cache.ehcache.EhCacheCacheManager">
  <property name="cacheManager" ref="ehcache" />
</bean>

<bean id="ehcache" class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean">
  <property name="configLocation" value="classpath:ehcache.xml" />
  <property name="shared" value="true" />
</bean>

<!-- Example endpoints beans -->
<bean id="dragonBallUserService" class="com.nicobrest.kamehouse.service.DragonBallUserService">
  <property name="dragonBallUserDao" ref="dragonBallUserDaoJpa" />
</bean>

<bean id="dragonBallUserDaoJpa" class="com.nicobrest.kamehouse.dao.DragonBallUserDaoJpa">
  <property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>
...

DragonBallUserDaoJpaTest.java

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations =  "classpath:applicationContext.xml" )
public class DragonBallUserDaoJpaTest 

private static final Logger LOGGER 
LoggerFactory.getLogger(DragonBallUserDaoJpaTest.class);

@Autowired
@Qualifier("dragonBallUserDaoJpa")
private DragonBallUserDaoJpa dragonBallUserDaoJpa;
...
@Test
public void createDragonBallUserTest() 

  DragonBallUser dragonBallUser = new DragonBallUser(null, "vegeta", "vegeta@dbz.com", 49, 40,
      1000);

  try 
    assertEquals(0, dragonBallUserDaoJpa.getAllDragonBallUsers().size());
    dragonBallUserDaoJpa.createDragonBallUser(dragonBallUser);
    assertEquals(1, dragonBallUserDaoJpa.getAllDragonBallUsers().size());
    dragonBallUserDaoJpa
        .deleteDragonBallUser(dragonBallUserDaoJpa.getDragonBallUser("vegeta").getId());
   catch (KameHouseBadRequestException | KameHouseNotFoundException e) 
    e.printStackTrace();
    fail("Caught unexpected exception.");
  

...

DragonBallUserDaoJpa.java

public class DragonBallUserDaoJpa implements DragonBallUserDao 

@Autowired
private EntityManagerFactory entityManagerFactory;
...
@CacheEvict(value =  "getAllDragonBallUsersCache" , allEntries = true)
public Long createDragonBallUser(DragonBallUser dragonBallUser) 

  EntityManager em = getEntityManager();
  try 
    em.getTransaction().begin();
    em.persist(dragonBallUser);
    em.getTransaction().commit();
   catch (PersistenceException pe) 
    ...
   finally 
    em.close();
  
  return dragonBallUser.getId();

...

我不知道还能尝试什么,有什么想法吗?

我正在使用 Spring 4.2.4.RELEASE、Hibernate 5.1.0.Final、Hibernate JPA 1.0.0.Final、ehcache 2.9.0、JUnit 4.12。

这绝对与 ehcache 相关,因为注释注释可以在测试类中自动装配 DAO,并且单元测试通过但我尝试了几个小时并无法弄清楚。

谢谢!

【问题讨论】:

如果变量具有相同的名称或类路径中只有一个可用的 bean,则您可以开始删除不需要的限定符。尝试在 bean 的声明中使用名称。限定符使用名称而不是 id 一开始我并没有建议,但我建议使用类配置而不是 xml 配置。在运行前更容易发现错误 感谢您的回答!我尝试删除限定符,但没有奏效。自动装配到接口而不是类解决了这个问题。我实际上也打算将所有内容都移到类配置中,我想练习类和 xml 配置。 【参考方案1】:

第一个错误是因为您尝试注入到DragonBallUserDaoJpa 而不是DragonBallUserDao。为了能够添加缓存层,Spring 在您的类上创建了一个代理。而这个代理实现类的接口(@98​​7654323@),然后委托给实际的类(DragonBallUserDaoJpa)。

您在检索 bean 时遇到了同样的问题。由于代理仅实现接口,因此您不能将其强制转换为实现。所以ClassCastException

所以如果你在测试中这样做(你不需要限定符)

@Autowired
private DragonBallUserDao dragonBallUserDao;

它应该可以解决问题。

另一个解决方案(但我认为它没有用)是强制 Spring 使用 cglib 创建代理。这样,您的代理确实会扩展具体的类。你需要这样的东西:&lt;aop:aspectj-autoproxy proxy-target-class="true" /&gt;

最后,你也可以去掉这个接口,因为我非常怀疑你有很多这个 DAO 的实现。所以界面没用,只会增加噪音。删除它会强制 Spring 创建一个 cglib 代理,因为没有可用的接口。

【讨论】:

太棒了!感谢你的回答!按照您的建议自动装配到接口而不是类。 @Autowired 私人 DragonBallUserDao dragonBallUserDaoJpa;我必须使用带有后缀 Jpa 的变量名,因为我对该接口有另一个实现。我绝对同意只有一个实现的接口会增加噪音而不是价值,但我实际上有 2 个实现,我将添加另一个带有 no-sql 数据库的实现。我使用这个项目主要是为了继续学习。我肯定需要更多地了解方面和代理。感谢您的帮助!【参考方案2】:

您的 junit 不会加载您的 spring conf xml。

试试这个:

@ContextConfiguration(locations = 
    "classpath:pathTo/applicationContext.xml")

编辑:尝试删除限定符和/或在您的 bean 创建中添加 name 属性

【讨论】:

对不起,我应该发布我的完整课程。它实际上确实加载了它。我正好有那个注释。它在向 DAO 添加 ehcache 注释之前工作并加载了 DAO。我用更多相关信息更新了课程

以上是关于将ehcache添加到DAO类后Junit测试失败,无法在测试类中实例化DAO的主要内容,如果未能解决你的问题,请参考以下文章

使用 DAO 和 Web 服务的数据库插入方法的 Junit 测试用例

HSQLDB Junit 测试对 DB2 和 SQL Server NOLOCK 查询失败

在JUNIT测试中没有创建Hibernate Transaction

如何将JUnit 4测试添加到JUnit 3测试套件中

如何在 JUnit4 中使用休眠测试 DAO

Junit测试调用Dao类的Business类