我的代码:
@Test
public void testAddRoleAndAddUser() {
Role r = roleDao.findByProperty("name", "admin");
if(r == null) {
r = new Role();
r.setName("admin");
r.setDescription("Just administrator.");
roleDao.save(r);
}
User u = dao.get(1l);
Set<Role> roles = u.getRoleSet();
logger.debug("Roles is null: " + (roles == null));
roles.add(r);
dao.save(u);
}
13:39:41,041 错误:org.hibernate.LazyInitializationException 无法延迟初始化角色集合:xxx.entity.core.User.roleSet,没有会话或会话已关闭 org.hibernate.LazyInitializationException:无法延迟初始化角色集合角色:xxx.entity.core.User.roleSet,在 org.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationException(AbstractPersistentCollection.java:380) 在 org.hibernate.collection.AbstractPersistentCollection.throwLazyInitializationExceptionIfNotConnected(AbstractPersistentCollection.java: 372) 在 org.hibernate.collection.AbstractPersistentCollection.initialize(AbstractPersistentCollection.java:365) 在 org.hibernate.collection.PersistentSet.add(PersistentSet.java:212) 在 sg.com.junglemedia.test.dao.impl.hibernate .UserDaoTest.testAddRoleAndAddUser(UserDaoTest.java:40) 在 sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 在 sun.reflect.NativeMethodAccessorImpl.invoke(NativeMet hodAccessorImpl.java:39) 在 sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25) 在 java.lang.reflect.Method.invoke(Method.java:597) 在 org.junit.runners.model.FrameworkMethod\(1。 org.junit.internal.runners.model.ReflectiveCallable.runReflectiveCall(FrameworkMethod.java:44) 在 org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41) 在 org .junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20) 在 org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28) 在 org.junit.runners.BlockJUnit4ClassRunner.runChild (BlockJUnit4ClassRunner.java:76) 在 org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50) 在 org.junit.runners.ParentRunner\)3.run(ParentRunner.java:193) 在 org.junit.runners.ParentRunner \(1.schedule(ParentRunner.java:52) 在 org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191) 在 org.junit.runners.ParentRunner.acce ss\)000(ParentRunner.java:42) 在 org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184) 在 org.junit.runners.ParentRunner.run(ParentRunner.java:236) 在 org.eclipse.jdt .internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:46) 在 org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) 在 org.eclipse.jdt.internal.junit .runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner .run(RemoteTestRunner.java:390) 在 org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)
有人帮忙吗?
原文由 Ben 发布,翻译遵循 CC BY-SA 4.0 许可协议
在您的实体类中,当您声明从用户到角色的映射时,请尝试将 fetchType 指定为 EAGER。像这样的事情:
更新:最近收到这个答案的评论让我重新审视这个。自从我回答以来已经有一段时间了,那时我才开始使用 Hibernate。拉斐尔和穆库斯说的有道理。如果你有一个大集合,你不应该使用预取。它联合选择映射到您的条目的所有数据并加载到内存中。另一种方法是仍然使用延迟获取并在每次需要处理相关集合时打开一个 Hibernate 会话,即每次需要调用 getRoleSet 方法时。这样,每次调用此方法时,Hibernate 都会执行对数据库的选择查询,并且不会将集合数据保存在内存中。详情可以参考我的帖子:http: //khuevu.github.io/2013/01/20/understand-hibernate.html
也就是说,它可能取决于您的实际用例。如果您的集合数据很小并且您经常需要查询数据,那么您最好使用预取。我认为,在您的具体情况下,角色集合可能非常小,适合使用急切获取。