如何在Hibernate Interceptor中获取Hibernate会话?

时间:2009-01-08 11:15:00

标签: java hibernate orm

如何在Hibernate Interceptor中获取Hibernate会话?

我正在尝试使用Hibernate透明地通过组织ID强制执行数据访问。 我已经设置了一个全局过滤器来按组织ID过滤所有查询。 现在,我需要使用Entity拦截器在保存/更新之前在所有实体上设置组织ID。

组织ID来自HttpSession

我在Hibernate会话中将Organizational Id设置为Filter属性,我希望在拦截器中检索它并用于所有Inserts和Updates。 问题是我似乎无法访问Interceptor中的Session。有没有解决方法呢?

4 个答案:

答案 0 :(得分:2)

如果您只需要组织ID,则可以将其放在静态ThreadLocal中,然后在拦截器中访问它。

另一方面,如果你已经准备好开始会话了,这取决于你的环境是什么,你可以放弃拦截器并使用org.hibernate.event.FlushEntityEventListener这似乎更符合你的想法反正需要。你可以像这样得到会话(粗略的伪代码):

FlushEntityEventListener.onFlushEntity(FlushEntityEvent event)
EntityEvent entityEvent = event.getEntityEntry();
EntityPersister persister = entityEvent.getPersister();
SessionFactoryImplementor sessionFactoryImplor = persister.getFactory();
Session session = sessionFactoryImplor.getCurrentSession();

来自Hibernate 3 On Line Docs事件系统可以另外使用或替代拦截器。

答案 1 :(得分:2)

你可以,但我会用一个简单的POJO来保持干净利落。请记住,存储在单例中的值只能由处理servlet请求的同一线程访问,因此如果您正在执行任何异步操作,则需要考虑该值。这是一个超级基本的印象:

public class OrgId {
   public static ThreadLocal<Integer> orgId = new ThreadLocal<Integer>();
}

由于组织ID驻留在会话中,您可以在这样的早期servlet过滤器中设置ThreadLocal的值(没有太多的错误检查):

public class OrgIdFilter implements Filter {
   public void doFilter(ServletRequest servletrequest, ServletResponse servletresponse, FilterChain filterchain) throws java.io.IOException, javax.servlet.ServletException {
      int orgId = 0;
      HttpServletRequest req = (HttpServletRequest) servletRequest;
      HttpSession session = req.getSession();
      orgId = Integer.parseInt(session.getAttribute("OrganizationalIdAttr"));
      try {
         OrgId.orgId.set(orgId);
         filterChain.doFilter(servletRequest, servletresponse);
      } finally {
         OrgId.orgId.set(null); // Important to clear after request !!
      }
   }
}

这假设orgId在调用过滤器时在会话中,但如果没有,你就会明白....

然后在你的拦截器(或几乎任何地方)你可以得到线程的当前orgId:

OrgId.orgId.get();   // Might be null.....

这里潜在的缺点是所有这些组件(过滤器,OrgId和拦截器)都需要由同一个类加载器加载,以确保OrgId类实际上是一个单例,否则,ThreadLocal的多个实例会挂在它周围不会一致地工作,或根本不工作。不用说,所有这些都需要在同一个VM中发生。

我不确定这是否是解决此问题的最简洁方法,但它确实可以让您在需要的地方找到它。

答案 2 :(得分:2)

创建拦截器时,如果可以为其提供SessionFactory的引用,则可以使用SessionFactory#getCurrentSession

答案 3 :(得分:0)

拦截器可以使BeanFactoryAware和SessionFactory可以使用可以从中获取当前会话的bean工厂获得。

由于循环依赖并使Interceptor知道Spring容器,因为它似乎是一个糟糕的设计,我使用了Nicholas建议的ThreadLocal