`

hibernate使用sql query时单元测试跑了一部分后就不动了。

阅读更多

我写了很多使用sql query的hibernate的单元测试,但是总是跑了一部分后就停止不前了。

由于特殊原因,我没能对session关闭,因为前人对它进行了封装,session.close()没有暴露,很恶心。

 

后来在数据库中mysql中,mysqladmin -uroot -pbb status,用这条命令查询连接,发现总是停在22个连接,我的测试用了20个连接,这些连接没有关闭。

 

暂时解决办法在hibernate配置文件中修改配置

<property name="minPoolSize" value="2"/>
<property name="maxPoolSize" value="100"/>

<!--把maxPoolSize由原来的20改成100-->

 

这样连接池的最大连接数增加了,测试是能通过了。

 

在我们的应用中,使用Spring的TransactionManager来管理事务,所以就可以放心在HibernateDaoSupport 中使用getSession (),释放的工作会有Spring帮你完成。

 

以下部分转载自:http://javachikuang.iteye.com/blog/261233

 

 

 

getSession()这个方法本身其实返回的是与当前事务绑定的Session对象,在HibernateDaoSupport中使 用,HibernateDaoSupport本身是不负责对这个Session对象进行关闭的,所以在其中有一个对应的releaseSession() 方法,用于关闭Session。

但是一般使用Spring时,我们会采用HibernateTransactionManager进行事务管理,把事务配置在Service层。 此时,它会帮助我们关闭与当前事务绑定的Session对象,这个可以参照HibernateTransactionManager类中的 doCleanupAfterCompletion方法,它是一个抽象方法的实现。再追溯上去,会发现,在事务commit或者rollback的时候, 会有一段finally代码,专门调用执行该方法的代码:

Java代码
  1. finally  {     
  2.             cleanupAfterCompletion(status);     
  3.         }     
  4.     
  5.     private   void  cleanupAfterCompletion(DefaultTransactionStatus status) {     
  6.         status.setCompleted();     
  7.         if  (status.isNewSynchronization()) {     
  8.             TransactionSynchronizationManager.clearSynchronization();     
  9.             TransactionSynchronizationManager.setCurrentTransactionName(null );     
  10.             TransactionSynchronizationManager.setCurrentTransactionReadOnly(false );     
  11.             if  (status.isNewTransaction()) {     
  12.                 TransactionSynchronizationManager.setActualTransactionActive(false );     
  13.             }     
  14.         }     
  15.         if  (status.isNewTransaction()) {     
  16.             doCleanupAfterCompletion(status.getTransaction());     
  17.         }     
  18.         if  (status.getSuspendedResources() !=  null ) {     
  19.             if  (status.isDebug()) {     
  20.                 logger.debug("Resuming suspended transaction" );     
  21.             }     
  22.             resume(status.getTransaction(), (SuspendedResourcesHolder) status.getSuspendedResources());     
  23.         }     
  24.     }    
finally {   
            cleanupAfterCompletion(status);   
        }   
  
    private void cleanupAfterCompletion(DefaultTransactionStatus status) {   
        status.setCompleted();   
        if (status.isNewSynchronization()) {   
            TransactionSynchronizationManager.clearSynchronization();   
            TransactionSynchronizationManager.setCurrentTransactionName(null);   
            TransactionSynchronizationManager.setCurrentTransactionReadOnly(false);   
            if (status.isNewTransaction()) {   
                TransactionSynchronizationManager.setActualTransactionActive(false);   
            }   
        }   
        if (status.isNewTransaction()) {   
            doCleanupAfterCompletion(status.getTransaction());   
        }   
        if (status.getSuspendedResources() != null) {   
            if (status.isDebug()) {   
                logger.debug("Resuming suspended transaction");   
            }   
            resume(status.getTransaction(), (SuspendedResourcesHolder) status.getSuspendedResources());   
        }   
    }  


故而,只要参与了事务,HibernateTransactionManager会帮你正确关闭Session。

不过很多的web应用都会采用OpenSessionInView模式,也就是Session会被保持到View层。同样经过 HibernateTransactionManager,为什么使用了OpenSessionInView模式以后,Session就不会被关闭呢?这 是由于在获取当前线程绑定事务的时候,有一个判断,如果存在当前线程绑定的Session,会把当前事务对象的newSessionHolder值设置成 false,从而跳过上面的代码中doCleanupAfterCompletion(status.getTransaction());的调用:

Java代码
  1. protected  Object doGetTransaction() {     
  2.         HibernateTransactionObject txObject = new  HibernateTransactionObject();     
  3.         txObject.setSavepointAllowed(isNestedTransactionAllowed());     
  4.     
  5.         if  (TransactionSynchronizationManager.hasResource(getSessionFactory())) {     
  6.             SessionHolder sessionHolder =     
  7.                     (SessionHolder) TransactionSynchronizationManager.getResource(getSessionFactory());     
  8.             if  (logger.isDebugEnabled()) {     
  9.                 logger.debug("Found thread-bound Session ["  +     
  10.                         SessionFactoryUtils.toString(sessionHolder.getSession()) + "] for Hibernate transaction" );     
  11.             }     
  12.             txObject.setSessionHolder(sessionHolder, false );     
  13.             if  (getDataSource() !=  null ) {     
  14.                 ConnectionHolder conHolder = (ConnectionHolder)     
  15.                         TransactionSynchronizationManager.getResource(getDataSource());     
  16.                 txObject.setConnectionHolder(conHolder);     
  17.             }     
  18.         }     
  19.     
  20.         return  txObject;     
  21.     }    
protected Object doGetTransaction() {   
        HibernateTransactionObject txObject = new HibernateTransactionObject();   
        txObject.setSavepointAllowed(isNestedTransactionAllowed());   
  
        if (TransactionSynchronizationManager.hasResource(getSessionFactory())) {   
            SessionHolder sessionHolder =   
                    (SessionHolder) TransactionSynchronizationManager.getResource(getSessionFactory());   
            if (logger.isDebugEnabled()) {   
                logger.debug("Found thread-bound Session [" +   
                        SessionFactoryUtils.toString(sessionHolder.getSession()) + "] for Hibernate transaction");   
            }   
            txObject.setSessionHolder(sessionHolder, false);   
            if (getDataSource() != null) {   
                ConnectionHolder conHolder = (ConnectionHolder)   
                        TransactionSynchronizationManager.getResource(getDataSource());   
                txObject.setConnectionHolder(conHolder);   
            }   
        }   
  
        return txObject;   
    }  


综合一下,只要使用Spring的TransactionManager来管理事务,就可以放心在HibernateDaoSupport中使用getSession(),释放的工作会有Spring帮你完成。

 

分享到:
评论

相关推荐

    Hibernate中的query 分页.doc

    不过不是每种数据库的都有这种分页支持的SQL,例如SQL Server就不支持。 4.scroll是利用JDBC2.0的功能做分页的,那么就完全取决于特定数据库的JDBC Driver的实现了。事实上大部分JDBC Driver都是把所有的结果集都...

    Hibernate+中文文档

    1.2. 第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第二部分 - 关联映射 1.3.1. ...

    HibernateAPI中文版.chm

    1.2. 第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第二部分 - 关联映射 1.3.1. ...

    hibernate3.2中文文档(chm格式)

    1.2. 第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第二部分 - 关联映射 1.3.1. ...

    Hibernate中文详细学习文档

    1.2. 第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第二部分 - 关联映射 1.3.1. ...

    Hibernate 中文 html 帮助文档

    1.2. 第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第二部分 - 关联映射 1.3.1. ...

    hibernate 体系结构与配置 参考文档(html)

    第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第二部分 - 关联映射 1.3.1. 映射...

    Hibernate_3.2.0_符合Java习惯的关系数据库持久化

    1.2. 第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第二部分 - 关联映射 1.3.1. ...

    Hibernate教程

    2.2. 第一部分 - 第一个Hibernate程序 2.2.1. 第一个class 2.2.2. 映射文件 2.2.3. Hibernate配置 2.2.4. 用Ant编译 2.2.5. 安装和帮助 2.2.6. 加载并存储对象 2.3. 第二部分 - 关联映射 2.3.1. 映射...

    Hibernate注释大全收藏

    一旦使用就不能使用AUTO和IDENTIFY生成器。 每个类层次结构一张表 @Entity @Inheritance(strategy=InheritanceType.SINGLE_TABLE) @DiscriminatorColumn( name="planetype", discriminatorType=...

    Hibernate参考文档

    1.2. 第一部分 - 第一个Hibernate应用程序 1.2.1. 第一个class 1.2.2. 映射文件 1.2.3. Hibernate配置 1.2.4. 用Ant构建 1.2.5. 启动和辅助类 1.2.6. 加载并存储对象 1.3. 第二部分 - 关联映射 1.3.1. ...

    hibernate3.04中文文档.chm

    2.2. 第一部分 - 第一个Hibernate程序 2.2.1. 第一个class 2.2.2. 映射文件 2.2.3. Hibernate配置 2.2.4. 用Ant编译 2.2.5. 安装和帮助 2.2.6. 加载并存储对象 2.3. 第二部分 - 关联映射 2.3.1. ...

    MyBatis第一天课堂笔记.docx

    1 Mybatis第一天课堂笔记 学习过的持久层框架:DBUtils , Hibernate Mybatis就是类似于hibernate的orm持久层框架。 为什么学Mybatis?...就算用hibernate的sqlquery,后续的维护工作也会让人发狂。

    hibernate 框架详解

    第一部分 - 第一个Hibernate程序 2.2.1. 第一个class 2.2.2. 映射文件 2.2.3. Hibernate配置 2.2.4. 用Ant编译 2.2.5. 安装和帮助 2.2.6. 加载并存储对象 2.3. 第二部分 - 关联映射 2.3.1. 映射...

    支持多数据库的ORM框架ef-orm.zip

    大部分OLTP应用系统到最后都不免要使用SQL/JPQL,然而没有一个很好的方法解决SQL在多种数据库下兼容性的问题。 EF-ORM中采用了独特的SQL解析和改写技术,能够主动检查并确保SQL语句或者SQL片段在各个数据库上的兼容...

    hibernate_reference中文文档.pdf

    1.1. 第一部分 - 第一个 Hibernate 应用程序 ................................. 1 1.1.1. 设置 ............................................................ 1 1.1.2. 第一个 class .............................

    经典JAVA.EE企业应用实战.基于WEBLOGIC_JBOSS的JSF_EJB3_JPA整合开发.pdf

     本书内容主要包括三部分,第一部分介绍java ee开发的基础知识,以及如何搭建开发环境,包括安装jboss、weblogic应用服务器,以及如何使用svn、netbeans等。第二部分详细讲解了jsf ri、jta、jndi、rmi、jms、...

    ibatis 开发指南(pdf)

    使用ibatis 提供的ORM 机制,对业务逻辑实现人员而言,面对的是纯粹的Java 对象, 这一层与通过Hibernate 实现ORM 而言基本一致,而对于具体的数据操作,Hibernate 会自动生成SQL 语句,而ibatis 则要求...

Global site tag (gtag.js) - Google Analytics