我的代码:
@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)
有人帮忙吗?
答案 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编组时。我已经使用了预取策略,但我也发现提供两个实体是有效的:
EAGER
公共字段,并在@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,您可以翻译。它还有一些参考文献。