我设置了@Transactional JUnit Test,我想将一些测试数据保存到数据库中,并测试关联是否正确。 但是,在测试关联时,它们总是评估为null,即使它在非事务性测试中有效。
我使用@Before注释坚持两个对象:
@Before
public void testData() {
TestObjectOne o = new TestObjectOne();
o.setName("One");
repo.saveEntity(o); //autowired
TestObjectTwo t = new TestObjectTwo();
t.setOne(o);
t.setName("Two");
repo.saveEntity(t);
}
在测试中访问这两个对象时,我得到了正确的实例:
TestObjectOne o = repo.getOneByName("One");
TestObjectOne t = repo.getTwoByName("Two");
检查t
和o
之间的关联时,我得到了正确的引用,因为我明确定义了该关联:
Assert.assertNotNull(t.getOne());
但是当反过来检查时,对象o
未正确更新:
Assert.assertNotNull(o.getTwos());
关联在域对象中定义为
在One
:
@OneToMany(mappedBy = "one", fetch = FetchType.EAGER)
private List<Two> twos;
在Two
:
@ManyToOne(optional = false)
private One one;
当我没有以@Transactional运行测试时,它的效果很好。
编辑 保存测试中的实体而不是@Before方法,没有任何区别。
答案 0 :(得分:4)
您需要在保存之后以及在加载断言的测试对象之前刷新并清除当前会话。
保存TestObjectOne o
时,它的实例保存在Hibernate会话中。比你创建TestObjectTwo t
并添加对o
的引用并保存它所以这个实例现在也保存在Hibernate会话中。比你调用时获取Hibernate获取你之前创建的实例而不更新它们以反映实际状态。所以你需要在加载之前刷新并清除会话 - 而不是会话L1缓存是空的,并且实体将被核心加载。
public class MyTest {
@Before
public void setUp() {
TestObjectOne o = new TestObjectOne();
o.setName("One");
repo.saveEntity(o); //autowired
TestObjectTwo t = new TestObjectTwo();
t.setOne(o);
t.setName("Two");
repo.saveEntity(t);
/* push all changes to current transaction*/
repo.getSessionFactory().getCurrentSession().flush();
/* clean current session, so when loading objects they'll
be created from sratch and will contain above changes */
repo.getSessionFactory().getCurrentSession().clear();
}
@Test
public void test() {
TestObjectOne o = repo.getOneByName("One");
TestObjectOne t = repo.getTwoByName("Two");
Assert.assertNotNull(t.getOne());
Assert.assertNotNull(o.getTwos());
}
}
答案 1 :(得分:0)
@Transactional注释在测试后触发回滚,因此结果不会持久存储到数据库中。
对于需要保留数据的测试,除了@Rollback(false)
之外,您还可以添加@Transactional
注释。