我从我的服务中调用dao
@Override
@Transactional
public Product getProductById(int id) {
return productDao.getProductById(id);
}
在dao我得到产品
@Override
public Product getProductById(int id) {
Product p = sessionFactory.getCurrentSession().load(Product.class, id);
System.out.print(p);
return p;
}
这样运行正常但如果我将我的dao类更改为
@Override
public Product getProductById(int id) {
return sessionFactory.getCurrentSession().load(Product.class, id);
}
我得到了org.hibernate.LazyInitializationException:无法初始化代理 - 没有Session。我正在打印产品的视图层中发生异常。我不明白为什么在dao方法中返回相同的行导致视图层中的异常,但如果我将它保存在引用中然后返回它,则工作正常。
答案 0 :(得分:5)
这是一个很好的reference,让您熟悉.get()和.load()方法的工作原理。
@Override
public Product getProductById(int id) {
Product p = sessionFactory.getCurrentSession().load(Product.class, id);
return p;
}
默认情况下, session.load()
返回代理对象而不会访问数据库。如果表上没有任何记录,它基本上返回NoObjectFoundError
,否则它将返回一个引用而不填充实际对象甚至命中数据库。
您的上述方法返回一个代理,因为它也必须初始化您的对象,会话保持打开状态并填充对象。
@Override
public Product getProductById(int id) {
return sessionFactory.getCurrentSession().load(Product.class, id);
}
但是在你的第二种方法中,基本上没有任何初始化就返回了代理。会议在此后关闭,无需事先使用。因此,你得到错误。
希望有所帮助
答案 1 :(得分:3)
此错误表示您尝试访问延迟加载的属性或集合,但hibernate会话已关闭或不可用。 Hibernate中的延迟加载意味着在代码中访问属性/集合之前,不会填充对象(通过数据库查询)。 Hibernate通过创建一个动态代理对象来实现这一点,该对象仅在您第一次使用该对象时才会访问数据库。为了实现这一点,您的对象必须在整个生命周期中附加到一个打开的Hibernate会话。
如果删除SOP语句,则根本不访问对象,因此未加载。当您尝试在其他部分代码中访问它时,它将抛出LazyInitializationException。
答案 2 :(得分:1)
当您处理Hibernate和查看图层时,这是一个典型的问题。发生错误是因为在呈现视图之前Hibernate会话已关闭。解决此问题的两种最简单方法是使用Open Session In View模式或在视图呈现之前获取视图中所需的所有数据。
由于您使用的是Spring,因此第一个解决方案是最简单的 - 只需应用OpenSessionInViewFilter
(如果您使用的是JPA,则为OpenEntityManagerInViewFilter
):
REPL http://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/orm/hibernate4/support/OpenSessionInViewFilter.html
您还应该了解使用OSIV模式的优缺点。
答案 3 :(得分:0)
在您的产品实体中,尝试在OneToMany关系注释中添加fetch = FetchType.EAGER,例如
@OneToMany(mappedBy =“ employee”,fetch = FetchType.EAGER)
这将加载整个Product对象图,避免后续调用