本人在项目中,实现分页查询时,遇到了这个问题,当多次翻页后,页面请求被挂起。项目使用spring MVC+hibernate。刷新页面没有反应,从新登陆同意不起作用,只有重启tomcat才能解决。经过1个多小时的调试,也没能找到问题所在,但发现了一个规律,就是每次出现页面卡死(即请求被挂起)都是在进行了固定次数(5)的翻页以后。由于程序未报任何异常,一直头疼中,但是突然想到了连接池,因为我的连接池最多连接数为10。想到了这里,感觉似乎抓到了问题的关键所在。然后百度了一下,果然就是这个原因。
问题代码如下:
protected <T> List<T> queryByPage(String sql,int page,int pageSize){ Session session = getSession(); Query query = session.createQuery(sql); //(page-1)*pageSize:将页码转化为实际数据库查询起始位置 query.setFirstResult((page-1)*pageSize); query.setMaxResults(pageSize); List<T> list = query.list(); return list; } protected String count(String sql){ Session session = getSession(); Query query = session.createQuery(sql); String count = query.list().get(0).toString(); return count; }
这里没有对session进行关闭和释放,因为我的代码每次都要获取分页数据和总数,所以刚好5次以后,连接池被耗尽。在return 语句前面加上releaseSession(session);问题解决。
另外还有一种方式可以解决该问题:为查询方法也使用事务(这里我使用的是注解方式:在方法前面加上@Transactional(propagation=Propagation.NEVER)),使用事务后spring就会为当前线程绑定session,在执行完本次请求之后,spring会自动释放session。代码应该修改为这样:
protected <T> List<T> queryByPage(String sql,int pageSize){ // Session session = getSession(); Session session = getSessionFactory().getCurrentSession(); Query query = session.createQuery(sql); //(page-1)*pageSize:将页码转化为实际数据库查询起始位置 query.setFirstResult((page-1)*pageSize); query.setMaxResults(pageSize); List<T> list = query.list(); // releaseSession(session); return list; }这里的session采用getSessionFactory().getCurrentSession();获得。问题得到解决。