问题 在Spring Transaction JUnit测试中自动装配Hibernate会话的正确方法


这个问题类似于以前的问题 。我在尝试着 @Autowire 我的一个Spring-JUnit-Transactional测试中的Hibernate会话,但是我得到了这个异常:

java.lang.IllegalStateException: No Hibernate Session bound to thread, and configuration does not allow creation of non-transactional ...

这是我的JUnit类:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"/applicationContext.xml"})
@TransactionConfiguration(transactionManager="transactionManager")
@Transactional
public class MyTest {
    @Qualifier("session")
    @Autowired
    private Session session;

    @Test
    public void testSomething() {
        session.get(User.class, "me@here.com");
    }
}

如果我,每一件工作都很好 @Autowire 一个 SessionFactory 得到我的 Session 以编程方式(而不是在Spring XML中定义它),如下所示:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"/applicationContext.xml"})
@TransactionConfiguration(transactionManager="transactionManager")
@Transactional
public class MyTest{    
    @Qualifier("sessionFactory")
    @Autowired
    private SessionFactory sessionFactory;

    @Test
    public void testSomething() {
    Session session = SessionFactoryUtils.getSession(sessionFactory, false);
        session.get(User.class, "me@here.com");
    }
}

但是,如果我定义我的话,我可以得到我原来的例子 Session 在我的Spring XML中 <aop:scoped-proxy /> 像这样:

<?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:tx="http://www.springframework.org/schema/tx"
        xmlns:aop="http://www.springframework.org/schema/aop"
        xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
        http://www.springframework.org/schema/tx
        http://www.springframework.org/schema/tx/spring-tx-2.5.xsd
        http://www.springframework.org/schema/aop 
        http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
        ">

    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
        ...
    </bean>

    <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="configLocation"><value>classpath:/hibernate.cfg.xml</value></property>
        <property name="configurationClass">
            <value>org.hibernate.cfg.AnnotationConfiguration</value>
        </property>
    </bean>

    <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
        <property name="sessionFactory" ref="sessionFactory"/>
        <property name="dataSource" ref="dataSource" />
    </bean>

    <tx:annotation-driven transaction-manager="transactionManager"/>

    <bean id="session" class="org.springframework.orm.hibernate3.SessionFactoryUtils" factory-method="getSession" scope="prototype">
        <constructor-arg ref="sessionFactory" />
        <constructor-arg value="false" />
        <!-- This is seems to be needed to get rid of the 'No Hibernate Session' error' -->
        <aop:scoped-proxy />
    </bean>
</beans>

我的问题是:为什么 <aop:scoped-proxy /> 鉴于我的单元测试中应该只有一个线程限制的事务上下文?什么  定义我的正确方法 Hibernate Session 豆?


9984
2017-09-30 21:32


起源



答案:


SessionFactoryUtils.getSession()与获取Session的任何其他方式一样好。 HibernateDaoSupport.getSession()会做同样的事情。

你需要scoped-proxy的原因是因为时间问题。如果没有scoped-proxy,它似乎是在测试开始之前注入Session,因此在事务开始之前注入会话,因此你会得到错误。

通过添加scoped-proxy,它代理Session并注入它,因此它不会预先注入实际会话(在事务开始之前),而是仅在测试运行时获取并调用它,当它实际需要创建一个时打电话给它。


6
2017-10-01 07:16





我认为“正确”的方式是注入 SessionFactory,并以编程方式从中获取会话。您获得异常的原因归结于记录的行为 SessionFactoryUtils.getSession()

获取给定的Hibernate会话   SessionFactory的。意识到并愿意   返回任何现有的对应   会话绑定到当前线程,   例如在使用时   HibernateTransactionManager的。将   否则,创建一个新的Session   “allowCreate”是真的。

由于没有任何内容将会话绑定到当前事务,因此失败。

我的建议是使用 HibernateTemplate  - 在您的上下文中定义一个,并将其自动装入您的测试中。 HibernateTemplate 与war会话具有大部分相同的操作,但会话处理位为您。你应该能够做到:

hibernateTemplate.get(User.class, "me@here.com");

4
2017-10-01 07:20



谢谢你的回复。如果我将“allowCreate”设置为true,则Spring似乎会创建第二个非事务性数据库会话,即@Transactional注释不会在测试期间回滚我的更改。自动装配HibernateTemplate的问题在于我有依赖于Session的DAO级别的类。我想我可以让它们依赖于HibernateTemplate,然后按照你的建议做一个get(User.class ...)。但是,我觉得我违反了Demeter法,因为DAO类的真正依赖是Session而不是HibernateTemplate。 - 0sumgain
您的DAO是注入Session还是SessionFactory?如果您正在注入会话,您可能想重新考虑一下,这可能不是一个好主意。 - skaffman
DAO注入Session。你能解释为什么那不是个好主意吗?谢谢。 - 0sumgain
因为Session是一个短暂的对象,并且您通常不会注入这样的短期对象。 - skaffman
再次感谢您的回复。是的,这是真的但是我的Session对象的范围是“请求”,Spring应该在每个HTTP请求结束时终止Session。 - 0sumgain


答案:


SessionFactoryUtils.getSession()与获取Session的任何其他方式一样好。 HibernateDaoSupport.getSession()会做同样的事情。

你需要scoped-proxy的原因是因为时间问题。如果没有scoped-proxy,它似乎是在测试开始之前注入Session,因此在事务开始之前注入会话,因此你会得到错误。

通过添加scoped-proxy,它代理Session并注入它,因此它不会预先注入实际会话(在事务开始之前),而是仅在测试运行时获取并调用它,当它实际需要创建一个时打电话给它。


6
2017-10-01 07:16





我认为“正确”的方式是注入 SessionFactory,并以编程方式从中获取会话。您获得异常的原因归结于记录的行为 SessionFactoryUtils.getSession()

获取给定的Hibernate会话   SessionFactory的。意识到并愿意   返回任何现有的对应   会话绑定到当前线程,   例如在使用时   HibernateTransactionManager的。将   否则,创建一个新的Session   “allowCreate”是真的。

由于没有任何内容将会话绑定到当前事务,因此失败。

我的建议是使用 HibernateTemplate  - 在您的上下文中定义一个,并将其自动装入您的测试中。 HibernateTemplate 与war会话具有大部分相同的操作,但会话处理位为您。你应该能够做到:

hibernateTemplate.get(User.class, "me@here.com");

4
2017-10-01 07:20



谢谢你的回复。如果我将“allowCreate”设置为true,则Spring似乎会创建第二个非事务性数据库会话,即@Transactional注释不会在测试期间回滚我的更改。自动装配HibernateTemplate的问题在于我有依赖于Session的DAO级别的类。我想我可以让它们依赖于HibernateTemplate,然后按照你的建议做一个get(User.class ...)。但是,我觉得我违反了Demeter法,因为DAO类的真正依赖是Session而不是HibernateTemplate。 - 0sumgain
您的DAO是注入Session还是SessionFactory?如果您正在注入会话,您可能想重新考虑一下,这可能不是一个好主意。 - skaffman
DAO注入Session。你能解释为什么那不是个好主意吗?谢谢。 - 0sumgain
因为Session是一个短暂的对象,并且您通常不会注入这样的短期对象。 - skaffman
再次感谢您的回复。是的,这是真的但是我的Session对象的范围是“请求”,Spring应该在每个HTTP请求结束时终止Session。 - 0sumgain