首页 > 代码库 > hibernate的延迟加载及其与session关闭的矛盾
hibernate的延迟加载及其与session关闭的矛盾
延迟加载就是并不是在读取的时候就把数据加载进来,而是等到使用时再加载。
那么Hibernate是怎么知道用户在什么时候使用数据了呢?又是如何加载数据呢?
其实很简单,它使用了代理机制。返回给用户的并不是实体本身,而是实体对象的代理。代理对象在用户调用getter方法时就会去数据库加载数据。
但加载数据就需要数据库连接。而当我们把会话关闭时,数据库连接就同时关闭了。这种情况就叫做未初始化的关系。
延迟加载与session关闭的矛盾一般可以这样处理:
1)、关闭延迟加载特性。 操作起来比较简单,因为hibernate的延迟加载特性是在hbm配置里面可控制的。默认lazy="true",改为lazy="false"就可以了。 但是使用这个解决办法带来的隐患是十分大的。 首先,出现no session or session was closed就证明了您已经在使用外键关联表,如果去掉延迟加载的话,则表示每次查询的开销都会变得十分的大,如果关联表越多,后果也可以想象得到。所以不建议使用这个方法解决。
2)、在session关闭之前把我们想要查询的数据先获取了。 首先需要了解一下session什么时候关闭,也就是它的生命周期。通常情况下hibernate会在查询数据关闭session,而使用getHibernateTemplate().get方法查询后会延迟关闭的时间。会在事务结束后才关闭。 使用拦截器(Interceptor)或过滤器(Filter)控制session。 spring为解决hibernate这一特性提供的解决方案,可以有效的控制session生命周期。
因为还没有讲到Spring,课堂上用过滤器(Filter)来控制session。
首先建个类OpenSessionInViewFilter继承Filter,重写doFilter方法,在try,finally中编写session的获得和关闭。实例代码如下:
public void doFilter( ServletRequest arg0, ServletResponse arg1, FilterChain chain) throws IOException, ServletException { try { HibUtil.getSession(); chain.doFilter(arg0, arg1); } finally { HibUtil.closeSession(); }}
然后在web.xml配置文件中增加相关配置,例如:
<filter> <filter-name>OpenSessionInViewFilter</filter-name> <filter-class> tarena.util.OpenSessionInViewFilter </filter-class></filter><filter-mapping> <filter-name>OpenSessionInViewFilter</filter-name> <url-pattern>*.action</url-pattern> </filter-mapping>
最后要记得修改service层事务处理的代码,把关闭session的代码注释掉。
hibernate的延迟加载及其与session关闭的矛盾