关于在 Spring 中对 DAO 进行单元测试的一些问题

最近在使用 Spring 2.0 和 Hibernate 3.2.0 进行开发,在对 DAO 进行单元测试的时候,出现了一些问题,因为对新环境不太熟悉,折腾了很久才把问题略为妥善的解决。

 

程序员喜欢用代码说话,所以先将测试的相关代码展示如下:

 

public class FilterSetDaoTest extends TestCase {

    private FilterSetDao filterSetDao;


[www.iocblog.net 来源]

 

    public void testCreateFilterSet() {

       FilterSet filterSet = new FilterSet();

       filterSet.setName("test10");

       filterSet.setCreateTime(new Date());

 

       Set<Filter> filters = new HashSet<Filter>();

       Filter filter = new Filter();

       filter.setRule("testrule10");

       filter.setType(FilterType.PLAIN);

       filter.setCreateTime(new Date());

       filters.add(filter);

       filterSet.setFilters(filters);

       filterSet.setUpdateTime(new Date());


 

       filterSetDao.saveOrUpdate(filterSet);


 

       FilterSet persistedFilterSet = filterSetDao.find(filterSet.getId());

[www.iocblog.net 来源]

       assertEquals(filterSet, persistedFilterSet);

       assertEquals(1, persistedFilterSet.getFilters().size());

    }

}


 

public interface FilterSetDao {

    public FilterSet find(int id);

}


 

public class FilterSetDaoHibernateImpl extends HibernateDaoSupport implements FilterSetDao {

    public FilterSet find(int id) {

       return (FilterSet) getHibernateTemplate().load(FilterSet.class,  id);

    }

}


 

@Entity

@Table(name = "filterset")

public class FilterSet implements Serializable {

    private Set<Filter> filters;


    @ManyToMany(cascade = CascadeType.ALL)

    public Set<Filter> getFilters() {

       return filters;

    }

}

 

因为在 Hibernate 中,Session.load() 方法返回的是实体类的一个代理类的实例,而此时因为没有合适的事务处理代码,相应的 session 已经关闭,所以在执行第一条 assertEquals() 方法时,即报告“LazyInitializationException: could not initialize proxy-the owning Session was closed”异常。可以将 DAO 实现中的 load() 换成 get()。

return (FilterSet) getHibernateTemplate().get(FilterSet.class, id);

因为 get() 是返回的不是代理类的实例,而直接返回实体类的实例,所以上面的异常将不会出现。但是 FilterSet 中的 filters 属性为 Set 类型,而在 Hibernate 3 中,默认对所有 Collection 和 Map 都采用 lazy initialization 的方式,因此在这里,基于上面所说的原因,又会报告“org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: net.patrickhe.FilterSet.filters, no session or session was closed” 这样的错误。所以,将 load() 换成 get() 其实并不是良好的解决办法。

按照 Spring Reference 上的说明,如果想在事务支持的环境下进行单元测试,可以让自己的测试用例类继承 org.springframework.test.annotation.AbstractAnnotationAwareTransactionalTests,如此即可。AbstractAnnotationAwareTransactionalTests 默认会关闭自动提交的特性,在测试方法执行完毕之后即进行回滚操作,以便清除方才测试时对数据库造成的修改变更。如果需要将测试数据提交到数据库中的话,那也很容易,直接调用 继承而来的 setComplete() 方法即可。关于 AbstractAnnotationAwareTransactionalTests 的更加详尽说明可以参考 Spring Reference - 8.3.3 Transaction management 一节。

 

注:在需要事务支持的环境中进行开发,如果使用 MySQL 作为持久化介质,一定要采用 InnoDB 之类的支持事务管理的存储引擎,否则一定会出现各种奇怪的逻辑错误。




文章整理:iocblog
版权申明:本站文章均来自网络,如有侵权,请联系我们,我们收到后立即删除,谢谢!
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有。