Vanilla JDBC更新未在弹簧单元测试中回滚

时间:2017-05-05 09:07:10

标签: java spring unit-testing junit spring-test

我有一个DAO方法的弹簧单元测试,它插入两个不同的表中。测试完成后,其中一个插入按预期回滚,但不是另一个!我真的无法弄清楚发生了什么。我已经多次调试了测试,以便我可以看到(未提交的)更改何时出现在数据库中,但只有一个消失了。

我能看到的唯一区别是,在一种情况下,插入是使用原始JDBC完成的,在第二种情况下是使用Sping的JdbcTemplate。但是,它们不应该在同一个事务中,然后回滚吗?

这是我的测试课程:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:spring/test-context.xml"})
@Rollback
@Transactional(transactionManager = "txManager")
public class MyDaoIntegrationTest {
    @Autowired
    private DataSource dataSource;
    @Autowired
    private MyDao myDao;

    @Test
    public void createMyObject_test() throws Exception {
        MyObject myObject = new MyObject();

        myDao.createMyObject(myObject, 123L);
    }
}

我的test-context.xml如下所示:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
   xsi:schemaLocation="
 http://www.springframework.org/schema/beans 
 http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">

<bean id="myDao" class="my.package.myDao">
    <constructor-arg name="dataSource" ref="dataSource"/>
</bean>

<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
    <property name="url" value="${unit.test.db.url}"/>
    <property name="driverClassName" value="com.microsoft.sqlserver.jdbc.SQLServerDriver"/>
    <property name="username" value="${unit.test.db.user}"/>
    <property name="password" value="${unit.test.db.pass}"/>
</bean>

<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource"/>
</bean>

</beans>

要测试的方法大致看起来(删除连接关闭时的噪音等),所以:

  public void createMyObject(MyObject vo, long refId) throws SQLException {

    Connection cn = dataSource.getConnection();
    PreparedStatement ps = cn.prepareStatement("insert into MY_OBJECT (COL1, COL, ...)) values (?,?,...)");
    ps.setInt(1, vo.getCol1());
    ps.setInt(2, vo.getCol2());
         ...
    ps.executeUpdate();

    MyObjectEventVO event = new MyObjectEventVO();
    createMyObjectEvent(event);
}

public void createMyObjectEvent(MyObjectEventVO vo) throws DataAccessException {

    JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
    String updateSql =
            "insert into MY_OBJECT_EVENT(COL1, COL2, ...) "
                    + "values (?, ?, ...)";
    Object[] params = {vo.getCol1(), vo.getCol2(), ... };
    int[] types = {Types.INTEGER, Types.VARCHAR, ...};

    jdbcTemplate.update(updateSql, params, types);
 }

更新 我已经尝试注释掉对createMyObjectEvent(event)的调用,以便只发生一次插入。最终结果是相同的:第一个插入不回滚。因此,第一次插入的事务似乎存在问题。

更新2 我重构了第一个插件也使用JdbcTemplate然后一切正常!所以这个问题可以改为:如何让你在弹簧单元测试中回滚vanilla JDBC代码?

1 个答案:

答案 0 :(得分:1)

@Rollback需要Spring管理的交易才能运作。 参与Spring托管交易最简单的方法是使用JDBCTemplateDataSourceUtils。 两者都包含使用spring托管事务所需的代码。