Hibernate:懒得初始化一个角色集合,没有会话或会话被关闭

Ben*_*Ben 23 java hibernate

我的代码:

    @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);
}
Run Code Online (Sandbox Code Playgroud)

13:39:41041 ERROR: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)at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)at sun.reflect.DelegatingMethodAccessorImpl.在org.junit的org.junit.runners.model.FrameworkMethod $ 1.runReflectiveCall(FrameworkMethod.java:44)的java.lang.reflect.Method.invoke(Method.java:597)中调用(DelegatingMethodAccessorImpl.java:25). org.junit.runners.runners.statements.InvokeMethod.evaluate上的org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)中的internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15) 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)at org.在org.junit.runners.ParentRunner.access $ 000(ParentRunner.java:42)的org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)上的junit.runners.ParentRunner $ 1.schedule(ParentRunner.java:52) org.junit.runners.ParentRunner $ 2.evaluate(ParentRunner.java:184)位于org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference的org.junit.runners.ParentRunner.run(ParentRunner.java:236).在运行org.eclipse.jdt.internal.junit.runner.RemoteTestRunner(JUnit4TestReference.java:46)在org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38).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)

有人帮吗?

Khu*_* Vu 31

在您的实体类中,当您声明从用户到角色的映射时,请尝试将fetchType指定为EAGER.有点像这样:

@OneToMany(fetch=FetchType.EAGER)
public Collection<Role> getRoleSet(){
 ...
}
Run Code Online (Sandbox Code Playgroud)

更新:最近的评论这个答案收到让我重新审视.我回答的时候已经有一段时间了,当时我才开始使用Hibernate.拉斐尔和穆库斯说的是合理的.如果你有一个大集合,你不应该使用渴望获取.它共同选择映射到您的条目的所有数据并加载到内存.另一种方法是,每次需要处理相关集合时,仍然使用延迟提取和打开Hibernate会话,即每次需要调用getRoleSet方法时.这样,每次调用此方法时,Hibernate都会对数据库执行select查询,并且不会将集合数据保留在内存中.有关详细信息,请参阅我的帖子:http://khuevu.github.io/2013/01/20/understand-hibernate.html

话虽如此,它可能取决于您的实际用例.如果您的收集数据很小并且您经常需要查询数据,那么最好使用急切提取.我认为,在您的具体情况下,角色的集合可能非常小,适合使用渴望获取.

  • @拉斐尔+1。这不能成为解决方案。当父级有多个 OneToMany 或其中任何一个返回相当多的行时,则不会。当您只需要来自一个父表的数据时,为什么有人想要从其他表返回数据。这是一个糟糕的糟糕的解决方案。我唯一想要使用 EAGER 的情况是它是否与此相反并且仅意味着一行(前提是这不会急切地加载其他加载其他内容的内容)。 (4认同)

Dra*_*onZ 17

您可以尝试将@Transactional注释添加到您的bean或方法中(如果所有变量的声明都放在方法中).


Jav*_*mae 5

您很可能会关闭 RoleDao 内的会话。如果您关闭会话然后尝试访问延迟加载的对象上的字段,您将收到此异常。您可能应该在测试中打开和关闭会话/事务。