如何在Hibernate Interceptor中获取Hibernate会话?
我正在尝试使用Hibernate透明地通过组织ID强制执行数据访问。 我已经设置了一个全局过滤器来按组织ID过滤所有查询。 现在,我需要使用Entity拦截器在保存/更新之前在所有实体上设置组织ID。
组织ID来自HttpSession
我在Hibernate会话中将Organizational Id设置为Filter属性,我希望在拦截器中检索它并用于所有Inserts和Updates。 问题是我似乎无法访问Interceptor中的Session。有没有解决方法呢?
答案 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