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

时间:2010-08-19 05:47:55

标签: 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);
}
  

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(母语   方法)at   sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)     在   sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)     在   java.lang.reflect.Method.invoke(Method.java:597)     在   org.junit.runners.model.FrameworkMethod $ 1.runReflectiveCall(FrameworkMethod.java:44)     在   org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)     在   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.access $ 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)

有人帮忙吗?

11 个答案:

答案 0 :(得分:29)

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

@OneToMany(fetch=FetchType.EAGER)
public Collection<Role> getRoleSet(){
 ...
}

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

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

答案 1 :(得分:13)

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

答案 2 :(得分:4)

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

答案 3 :(得分:4)

我遇到了同样的问题,所以只是在我调用DAO方法的地方添加了@Transactional注释。它只是有效。我认为问题是Hibernate不允许从数据库中检索子对象,除非在调用时特别是所有必需的对象。

答案 4 :(得分:3)

以下代码可能会导致类似错误:

  using (var session = SessionFactory.OpenSession())
  using (var tx = session.BeginTransaction())
  {
      movie = session.Get<Movie>(movieId);
      tx.Commit();
  }
  Assert.That(movie.Actors.Count == 1);

你可以简单地解决它:

  using (var session = SessionFactory.OpenSession())
  using (var tx = session.BeginTransaction())
  {
      movie = session.Get<Movie>(movieId);
      Assert.That(movie.Actors.Count == 1);
      tx.Commit();
  }

答案 5 :(得分:3)

对我而言,它也适用于我在eclipselink中使用的方法。在将其用作页面参数之前,只需调用应加载的集合的size()。

for (Entity e : entityListKeeper.getEntityList()) {
    e.getListLazyLoadedEntity().size();
}

这里entityListKeeper具有包含LazyLoadedEntity列表的实体列表。 如果你只有那个实体有LazyLoadedEntity列表,那么解决方案是:

getListLazyLoadedEntity().size();

答案 6 :(得分:2)

你有不同的选择来处理这个问题。它似乎把它带回了旧的好的SQL日期:)

阅读本文:http://www.javacodegeeks.com/2012/07/four-solutions-to-lazyinitializationexc_05.html

答案 7 :(得分:2)

我遇到过这个问题,特别是当实体被Jaxb + Jax-rs编组时。我已经使用了预取策略,但我也发现提供两个实体是有效的:

  1. 将所有集合映射为EAGER
  2. 的完整实体
  3. 整理了大部分或全部收藏品的简化实体
  4. 公共字段,并在@MappedSuperclass中映射,并由两个实体实现进行扩展。

    当然,如果您总是需要加载集合,那么没有理由不EAGER加载它们。就我而言,我希望实体的精简版本能够在网格中显示。

答案 8 :(得分:1)

在我的情况下,Exception发生是因为我删除了 &#34; hibernate.enable_lazy_load_no_trans =真&#34;在&#34; hibernate.properties&#34;文件...

我做了一个复制并粘贴错误...

答案 9 :(得分:0)

您尝试加载延迟加载的集合,但hibernate会话已关闭或不可用。 对于此问题的最佳解决方案,将延迟加载的对象更改为eager fetch = FetchType.EAGER加载。这个问题会解决。

答案 10 :(得分:-3)

以西班牙语检查this blog post,您可以翻译。它还有一些参考文献。