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

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

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

组织ID来自HttpSession

我在Hibernate会话中将Organizational Id设置为Filter属性,我希望在拦截器中检索它并用于所有Inserts和Updates。 问题是我似乎无法访问拦截器内的Session。 这有什么变通方法吗?

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

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

由于组织ID驻留在会话中,您可以像这样在早期的servletfilter中设置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在调用filter时在会话中,但如果没有,你会得到这个想法….

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

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

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

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

如果您只需要组织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 : 事件系统可以作为拦截器的补充或替代使用。

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

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

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