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

时间:2009-09-30 21:32:57

标签: java hibernate spring transactions junit

此问题类似于之前的one。我在我的一个Spring-JUnit-Transactional测试中试图@Autowire一个Hibernate Session但是我得到了这个例外:

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 bean的正确方法?

2 个答案:

答案 0 :(得分:6)

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

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

通过添加scoped-proxy,它代理Session并注入它,因此它不会提前注入实际会话(在事务开始之前),但只有在测试运行时才会获取它并在它实际需要时对其进行调用打个电话。

答案 1 :(得分:4)

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

  

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

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

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

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