Hibernate为我们封装了对数据库的操作,可以通过实体映射来关联数据库。在加载数据时,Hibernate对关联对象、属性进行延迟加载,延迟加载可以提升性能,使用的时候才进行加载。但是这样就存在一个问题,必须保证延迟加载的操作限于同一个 Hibernate Session 范围之内进行。如果 Service 层返回一个启用了延迟加载功能的领域对象给 Web 层,当 Web 层访问到那些需要延迟加载的数据时,由于加载领域对象的 Hibernate Session 已经关闭,这些导致延迟加载数据的访问异常。如下:
一种解决方式是,将映射文件中的延迟加载属性设置为layz="false",即去掉延迟加载。这样我们会失去延迟加载带来的作用于优势,显然这种方式是不友好的。
为解决这与问题,Spring提供了OpenSessionInViewFilter过滤器,主要功能是用来把一个Hibernate Session和一次完整的请求过程对应的线程相绑定。它允许在事务提交之后延迟加载显示所需要的对象。
在web.xml文件进行如下配置:
<!-- 配置解决懒加载问题 :懒加载,在事务提交之后session关闭,造成页面无法获取session的值,而不能进行回显-->
<filter>
<filter-name>OpenSessionInViewFilter</filter-name>
<filter-class>org.springframework.orm.hibernate3.support.OpenSessionInViewFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>OpenSessionInViewFilter</filter-name>
<!-- 配置在执行所有action之后执行该filter,对session进行管理 -->
<url-pattern>*.action</url-pattern>
</filter-mapping>
小结:
OpenSessionInViewFilter 过滤器将 Hibernate Session 绑定到请求线程中,它将自动被 Spring 的事务管理器探测到。所以 OpenSessionInViewFilter 适用于 Service 层使用HibernateTransactionManager 或 JtaTransactionManager 进行事务管理的环境,也可以用于非事务只读的数据操作中。