2

我有很多关于休眠会话的问题要问,因为我一直遇到问题。

我正在使用 Spring 3.1.1 和 Hibernate 4.1.3。

首先是我正在处理的类图。

类图

代码类:

public class Equipement {

  @ManyToOne(fetch = FetchType.LAZY)
  @Cascade({CascadeType.SAVE_UPDATE, CascadeType.DELETE})
  public getOffice(){
    return office ;
  }

public class Office {    
  @ManyToOne(fetch = FetchType.LAZY)
  @Cascade({CascadeType.SAVE_UPDATE, CascadeType.DELETE})
  public getService(){
    return service ;
  }

public class Service {
  @ManyToOne(fetch = FetchType.LAZY)
  @Cascade({CascadeType.SAVE_UPDATE, CascadeType.DELETE})
  public getDepartement(){
    return departement ;
  }

这是我加载所有设备的方法。

public class HibernateEquipementDao{

  @SuppressWarnings("unchecked")
  public List<Equipement> getAll() {
      return sessionFactory.getCurrentSession().createQuery("from Equipement").list() ;
  }
}

假设我将所有设备加载到 a 中List<Equipement> allEquipements,并且希望仅使用某些服务或某些部门的设备来填充我的视图。

 List<Equipement> aListOfEquipements = new ArrayList<Equipement>() ;
 for(Equipement equipement : allEquipements)
   if(equipement.getOffice().getService().getName().equals("name"))
     aListOfEquipements.add(equipement) ;

我真的可以随时这样做吗?名为Current的会话始终处于活动状态?有时我会得到那个例外。 org.hibernate.LazyInitializationException:无法初始化代理 - 没有会话

这让我问我的 EquipementDao 使用的当前会话在哪里?

当我想获得设备部门时,有没有办法打开会话

还是有另一种方法可以继续?如何管理会话以及如何加载我的所有设备并毫无问题地访问其他实体?

4

3 回答 3

3

为服务添加层,我知道这个名字在你的情况下不是那么幸运,但是在这个层中你将有负责你的逻辑的类。

一些原型:

public class EquipmentService {

   @Inject
   private HibernateEquipementDao dao;

   @Transactional
   public List<Equipement> doSomethingWithEquipmentWithoutLazyException(Equipment equipment){
      List<Equipement> allEquipments = dao.getAll();

      List<Equipement> aListOfEquipements = new ArrayList<Equipement>() ;
      for(Equipement equipement : allEquipements) {
         if(equipement.getOffice().getService().getName().equals("name")) {
             aListOfEquipements.add(equipement) ;
         }
      }
      return aListOfEquipments;
   }

}

非常重要的是注解:@Transactional这意味着会话将在方法开始时打开并在方法结束时关闭,因此无法抛出 LazyLoadingException。

关于 Transactonal 的一些有趣的话题:@Transactional 注释属于哪里?

于 2012-09-24T12:30:55.290 回答
3

我已经看到了两种解决这个问题的方法。

1.在视图中打开会话

您可以在视图中配置您的休眠设置打开会话。如果你以编程方式配置你的休眠,你可以使用这一行:

jpaProperties.put(org.hibernate.cfg.Environment.ENABLE_LAZY_LOAD_NO_TRANS, true)

另一种方法是在 web.xml 中声明一个过滤器,如下所示:

    <filter>
        <filter-name>OpenEntityManagerInViewFilter</filter-name>
        <filter-class>org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter</filter-class>
    </filter>

    <filter-mapping>
        <filter-name>OpenEntityManagerInViewFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>

这种方法会影响您的所有系统,并且可能对您的性能不利。

2.获取加入:

您可以在查询中使用 FETCH JOIN 来强制休眠加载数据,例如:

from Equipement e FETCH JOIN e.office

这种方法将更难做(您的系统中的许多点受到影响)并且要小心。

3、查询返回后调用get方法:

在您的 DAO 中,您可以在查询返回后调用 get 方法。我不喜欢这种方式,但通常工作。

嗯,我希望我能帮助你。

于 2012-11-29T19:07:13.347 回答
0

对于大多数 Web 应用程序来说,让当前事务跨越整个请求是最简单且绝对合适的。最简单的方法是有一个过滤器来启动事务并在请求结束时提交/回滚。在 Spring/Hibernate 中,这称为 Open Entity in View 模式。Spring 通过 org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter 为此提供了一个过滤器实现。

当使用这种方法时,视图,即您的 JSP,参与相同的事务并且不需要重新加载,并且在 fetchtype LAZY 关系的情况下也可以导航实体。

于 2012-09-24T20:32:55.747 回答