Session is closed; nested exception is org.hibernate.SessionException: Session is closed解决方案

本文主要是介绍Session is closed; nested exception is org.hibernate.SessionException: Session is closed解决方案,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本人错误在于使用spring管理事务后,还手动关闭session。

1.通过getSession()方法获得session进行操作 

Java代码
  1. public class Test  extends HibernateDaoSupport{   
  2.      public void save(User user){   
  3.         this.getSession().save(user);   
  4.      }   
  5. }    
[java]  view plain copy
  1. public class Test  extends HibernateDaoSupport{  
  2.      public void save(User user){  
  3.         this.getSession().save(user);  
  4.      }  
  5. }    

利用这种方式获得的session在方法执行结束之后不会自动关闭连接,也就是说我们必须通过session.close()或者releaseSession(session)来手动进行关闭,否则会造成内存泄露或者连接耗尽等问题。手动关闭:  
Java代码
  1. public class Test  extends HibernateDaoSupport{   
  2.      public void save(User user){   
  3.         Session session = this.getSession();   
  4.         session.save(user);   
  5.         session.close();   
  6.        
  7.        // releaseSession(session); //这种方法,我用过的,绝对能解决session.close()的错误。
  8.      }   
  9. }   
[java]  view plain copy
  1. public class Test  extends HibernateDaoSupport{  
  2.      public void save(User user){  
  3.         Session session = this.getSession();  
  4.         session.save(user);  
  5.         session.close();  
  6.         // releaseSession(session);   
  7.      }  
  8. }   

如果对上述方法进行事务控制,那么spring框架会自动为我们关闭session,此种情况( session.close() )下再执行上述代码,会抛出如下异常:
Java代码
  1.  org.springframework.orm.hibernate3.HibernateSystemException: Session is closed; nested exception is org.hibernate.SessionException: Session is closed   
  2. …   
  3. org.hibernate.SessionException: Session is closed  
[java]  view plain copy
  1.  org.springframework.orm.hibernate3.HibernateSystemException: Session is closed; nested exception is org.hibernate.SessionException: Session is closed  
  2. …  
  3. org.hibernate.SessionException: Session is closed  

提示session已经关闭。但是如果在代码中通过releaseSession(session)的方法来关闭session,则不会抛出异常。releaseSession(session)方法的代码如下:  
Java代码
  1. protected final void releaseSession(Session session) {   
  2.     SessionFactoryUtils.releaseSession(session, getSessionFactory());   
  3. }  
[java]  view plain copy
  1. protected final void releaseSession(Session session) {  
  2.     SessionFactoryUtils.releaseSession(session, getSessionFactory());  
  3. }  

也就是说它是通过SessionFactoryUtils的releaseSession方法来实现的:  
Java代码
  1. public static void releaseSession(    
  2.      Session session,SessionFactory sessionFactory) {   
  3.           if (session == null) {   
  4.               return;   
  5.           }   
  6.           // Only close non-transactional Sessions.   
  7.           if (!isSessionTransactional(session,sessionFactory))   {   
  8.              closeSessionOrRegisterDeferredClose  (session, sessionFactory);   
  9.           }   
  10.     }  
[java]  view plain copy
  1. public static void releaseSession(   
  2.      Session session,SessionFactory sessionFactory) {  
  3.           if (session == null) {  
  4.               return;  
  5.           }  
  6.           // Only close non-transactional Sessions.  
  7.           if (!isSessionTransactional(session,sessionFactory))   {  
  8.              closeSessionOrRegisterDeferredClose  (session, sessionFactory);  
  9.           }  
  10.     }  

可见它内部会先进行判断。  

查看getSession()方法的源码:  
Java代码
  1. protected final Session getSession()   
  2.         throws DataAccessResourceFailureException, IllegalStateException {   
  3.   
  4.         return getSession(this.hibernateTemplate.isAllowCreate());   
  5. }  
[java]  view plain copy
  1. protected final Session getSession()  
  2.         throws DataAccessResourceFailureException, IllegalStateException {  
  3.   
  4.         return getSession(this.hibernateTemplate.isAllowCreate());  
  5. }  

getSession()方法内部通过它的一个重载方法getSession(boolean allowCreate )来实现,变量allowCreate是HibernateTemplate中的变量,默认值为true,也就是创建一个新的session。如果我们调用getSession(false)来获得session,那么必须对其进行事务控制,原因是:(spring文档)
Java代码
  1. protected  final  org.hibernate.Session  getSession()    
  2. throws DataAccessResourceFailureException,   IllegalStateException     
  3.   
  4. Get a Hibernate Session, either from the current transaction or a new one. The latter is only allowed if the "allowCreate" setting of this bean's HibernateTemplate is true.   
[java]  view plain copy
  1. protected  final  org.hibernate.Session  getSession()   
  2. throws DataAccessResourceFailureException,   IllegalStateException    
  3.   
  4. Get a Hibernate Session, either from the current transaction or a new one. The latter is only allowed if the "allowCreate" setting of this bean's HibernateTemplate is true.   

也就是说,getSession()方法从当前事务或者一个新的事务中获得session,如果想从一个新的事务中获得session(也就意味着当其不存在事务控制),则必须使HibernateTemplate中的allowCreate变量的值为”true”,而现在设置allowCreate变量的值为”false”就意味着无法从新的事务中获得session,也就是只能从当前事务中获取,所以必须对当前方法进行事务控制,否则会抛出如下异常:  
Java代码
  1. java.lang.IllegalStateException: No Hibernate Session bound to thread, and configuration does not allow creation of non-transactional one here ...  
[java]  view plain copy
  1. java.lang.IllegalStateException: No Hibernate Session bound to thread, and configuration does not allow creation of non-transactional one here ...  

同时,如果对getSession()所在的方法进行事务控制,那么类似如下的代码:  
Java代码
  1. Session session = null;   
  2. for(int m =0;m<5;m++){   
  3.     Admin admin = new Admin();   
  4.     admin.setName("test");   
  5.     admin.setPassword("098");      
  6.     session = this.getSession();   
  7.     session.save(admin);   
  8. }  
[java]  view plain copy
  1. Session session = null;  
  2. for(int m =0;m<5;m++){  
  3.     Admin admin = new Admin();  
  4.     admin.setName("test");  
  5.     admin.setPassword("098");     
  6.     session = this.getSession();  
  7.     session.save(admin);  
  8. }  

只会打开一个session,因为事务控制必须确保是同一个连接,spring会确保在整个相关方法中只存在一个session。Spring在方法开始时会打开一个session(即使进行事务控制的方法内部不执行数据库操作),之后在请求session时,如果在事务中存在一个未commit的session就返回,以此确保同一个session。  

2.getCurrentSession()与openSession()  
getCurrentSession()与openSession()方法通过Hibernate的SessionFactory获得,两者的区别网上有很多文章已经介绍过,即:
Java代码
  1. ①getCurrentSession创建的session会和绑定到当前线程,而openSession不会。    
  2. ②getCurrentSession创建的线程会在事务回滚或事物提交后自动关闭,而openSession必须手动关闭  
[java]  view plain copy
  1. ①getCurrentSession创建的session会和绑定到当前线程,而openSession不会。   
  2. ②getCurrentSession创建的线程会在事务回滚或事物提交后自动关闭,而openSession必须手动关闭  


对于getCurrentSession()方法: 
        (1)其所在方法必须进行事务控制 
        (2)Session在第一次被使用的时候,或者第一次调用getCurrentSession()的时候,其生命周期就开始。然后它被Hibernate绑定到当前线程。当事务结束的时候,不管是提交还是回滚,Hibernate也会把Session从当前线程剥离,并且关闭它。假若你再次调用getCurrentSession(),你会得到一个新的Session,并且开始一个新的工作单元。     
   
对于openSession()方法: 
         这个方法一般在spring与Hibernate的集成中不直接使用,它就是打开一个session,并且这个session与上下文无关,如果对其所在方法进行事务控制,会发现不起作用,原因就是前面提到的,事务控制必须确保是同一个连接,而openSession()打开的session与上下文无关。这个方法与getSession(),getCurrentSession()以及getHibernateTemplate()等方法的区别在于:后面的几个方法spring可以对其进行控制,如果对它们所在的方法进行事务控制,spring可以确保是同一个连接,而openSession()方法,spring无法对其进行控制,所以事务也不会起作用。
 

3.OpenSessionInView  
OpenSessionInView的主要功能是用来把一个Hibernate Session和一次完整的请求过程对应的线程相绑定。Open Session In View在request把session绑定到当前thread期间一直保持hibernate session在open状态,使session在request的整个期间都可以使用,如在View层里PO也可以lazy loading数据,如 ${ company.employees }。当View 层逻辑完成后,才会通过Filter的doFilter方法或Interceptor的postHandle方法自动关闭session。  
Java代码
  1. public class Group implements Serializable{    
  2.     private int id;    
  3.     private String name;    
  4.     private Set users;   
  5.          ...   
  6. }  
[java]  view plain copy
  1. public class Group implements Serializable{   
  2.     private int id;   
  3.     private String name;   
  4.     private Set users;  
  5.          ...  
  6. }  

在业务方法中加载Group对象并将其保存到HttpSession对象中  
Java代码
  1. List groups = ht.find("from Group");   
  2. Group group = (Group)groups.get(0);   
  3. HttpSession session = ServletActionContext.getRequest().getSession();   
  4. session.setAttribute("group", group);  
[java]  view plain copy
  1. List groups = ht.find("from Group");  
  2. Group group = (Group)groups.get(0);  
  3. HttpSession session = ServletActionContext.getRequest().getSession();  
  4. session.setAttribute("group", group);  

注意Group采用默认的延迟加载机制,即此时返回的只是一个Group代理对象, 
在jsp页面中显示group对象的users属性,如下: 

Java代码
  1. <%     
  2.      Group group = (Group)session.getAttribute("group");   
  3.      out.println(group.getUsers());   
  4. %>   
[java]  view plain copy
  1. <%    
  2.      Group group = (Group)session.getAttribute("group");  
  3.      out.println(group.getUsers());  
  4. %>   

此时会抛出如下异常:  
Java代码
  1. org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: entity.Group.users, no session or session was closed  
[java]  view plain copy
  1. org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: entity.Group.users, no session or session was closed  

延迟加载机制使得在业务方法执行结束之后仅仅返回Group的一个代理对象,在jsp页面中使用到group对象的值时,才发出sql语句加载,但此时session已经关闭。解决方法是采用OpenSessionInView机制,在web.xml页面中配置如下过滤器:  
Java代码
  1. <filter>     
  2.    <filter-name>hibernateFilter</filter-name>    
  3.    <filter-class>    
  4. org.springframework.orm.hibernate3.support.OpenSessionInViewFilter   
  5.    </filter-class>     
  6. </filter>  
[java]  view plain copy
  1. <filter>    
  2.    <filter-name>hibernateFilter</filter-name>   
  3.    <filter-class>   
  4. org.springframework.orm.hibernate3.support.OpenSessionInViewFilter  
  5.    </filter-class>    
  6. </filter>  


总结:  
(1) 对于getSession(),getSession(false),getCurrentSession()以及getHibernateTemplate()方法而言,如果对其所在方法进行事务控制,那么可以确保在整个方法中只存在一个session,无论你执行了几次CRUD操作,并且所打开的session会在事务结束时自动关闭。 
(2) 必须对getSession(false)以及getCurrentSession()所在的方法进行事务控制(原因见上述分析) 
(3) 如果没有对getSession()以及getHibernateTemplate()所在方法进行事务控制,那么如果在方法中进行N次CRUD操作,就会打开N个session,即每次调用getSession()和getHibernateTemplate()方法都会打开新的session。这两个方法的区别在于:getHibernateTemplate()方法结束时会自动关闭连接,而getSession()方法必须手动关闭。 
(4) 如果在方法中采用SessionFactory的openSession()方法获得连接进行操作,那么无法对其进行事务控制。 
(5) 一般的开发中,通常采用getHibernateTemplate()方法进行数据库操作, getHibernateTemplate()方法采用模板+回调的机制,进行数据库操作很方便,可以查看(其中session的打开与关闭都是在doExecute方法中进行的)

这篇关于Session is closed; nested exception is org.hibernate.SessionException: Session is closed解决方案的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/955941

相关文章

使用Python构建智能BAT文件生成器的完美解决方案

《使用Python构建智能BAT文件生成器的完美解决方案》这篇文章主要为大家详细介绍了如何使用wxPython构建一个智能的BAT文件生成器,它不仅能够为Python脚本生成启动脚本,还提供了完整的文... 目录引言运行效果图项目背景与需求分析核心需求技术选型核心功能实现1. 数据库设计2. 界面布局设计3

Java.lang.InterruptedException被中止异常的原因及解决方案

《Java.lang.InterruptedException被中止异常的原因及解决方案》Java.lang.InterruptedException是线程被中断时抛出的异常,用于协作停止执行,常见于... 目录报错问题报错原因解决方法Java.lang.InterruptedException 是 Jav

kkFileView在线预览office的常见问题以及解决方案

《kkFileView在线预览office的常见问题以及解决方案》kkFileView在线预览Office常见问题包括base64编码配置、Office组件安装、乱码处理及水印添加,解决方案涉及版本适... 目录kkFileView在线预览office的常见问题1.base642.提示找不到OFFICE组件

SpringBoot监控API请求耗时的6中解决解决方案

《SpringBoot监控API请求耗时的6中解决解决方案》本文介绍SpringBoot中记录API请求耗时的6种方案,包括手动埋点、AOP切面、拦截器、Filter、事件监听、Micrometer+... 目录1. 简介2.实战案例2.1 手动记录2.2 自定义AOP记录2.3 拦截器技术2.4 使用Fi

WinForm跨线程访问UI及UI卡死的解决方案

《WinForm跨线程访问UI及UI卡死的解决方案》在WinForm开发过程中,跨线程访问UI控件和界面卡死是常见的技术难题,由于Windows窗体应用程序的UI控件默认只能在主线程(UI线程)上操作... 目录前言正文案例1:直接线程操作(无UI访问)案例2:BeginInvoke访问UI(错误用法)案例

Spring Security常见问题及解决方案

《SpringSecurity常见问题及解决方案》SpringSecurity是Spring生态的安全框架,提供认证、授权及攻击防护,支持JWT、OAuth2集成,适用于保护Spring应用,需配置... 目录Spring Security 简介Spring Security 核心概念1. ​Securit

MySQL逻辑删除与唯一索引冲突解决方案

《MySQL逻辑删除与唯一索引冲突解决方案》本文探讨MySQL逻辑删除与唯一索引冲突问题,提出四种解决方案:复合索引+时间戳、修改唯一字段、历史表、业务层校验,推荐方案1和方案3,适用于不同场景,感兴... 目录问题背景问题复现解决方案解决方案1.复合唯一索引 + 时间戳删除字段解决方案2:删除后修改唯一字

MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)

《MyBatis-Plus中nested()与and()方法详解(最佳实践场景)》在MyBatis-Plus的条件构造器中,nested()和and()都是用于构建复杂查询条件的关键方法,但... 目录MyBATis-Plus 中nested()与and()方法详解一、核心区别对比二、方法详解1.and()

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

全面解析MySQL索引长度限制问题与解决方案

《全面解析MySQL索引长度限制问题与解决方案》MySQL对索引长度设限是为了保持高效的数据检索性能,这个限制不是MySQL的缺陷,而是数据库设计中的权衡结果,下面我们就来看看如何解决这一问题吧... 目录引言:为什么会有索引键长度问题?一、问题根源深度解析mysql索引长度限制原理实际场景示例二、五大解决