在Spring整合 Hibernate中,对dao层访问中提供了两个操作。
(1)protected final Session getSession() throws DataAccessResourceFailureException,IllegalStateException;
spring api的解释:
Obtain a Hibernate Session, either from the current transaction or a new one. The latter is only allowed if the "allowCreate" setting of this bean'sHibernateTemplate is "true".
Note that this is not meant to be invoked from HibernateTemplate code but rather just in plain Hibernate code. Either rely on a thread-bound Session or use it in combination with releaseSession(org.hibernate.Session).
In general, it is recommended to use HibernateTemplate, either with the provided convenience operations or with a custom HibernateCallback that provides you with a Session to work on. HibernateTemplate will care for all resource management and for proper exception conversion.
(2)public final HibernateTemplate getHibernateTemplate();
spring api的解释:
Note: The returned HibernateTemplate is a shared instance.
通过http://blog.youkuaiyun.com/fenixshadow/archive/2007/09/26/1802277.aspx的分析:可知getHibernateTemplate由spring管理,始终使用一个session连接数据库,而每次getSession()就会创建一个新的session连接数据库,不被spring管理。
转帖(一)http://blog.youkuaiyun.com/fenixshadow/archive/2007/09/26/1802277.aspx
都说Spring 和 Hibernate是绝配。今天有点小小心得,拿出来共享一下。
当我们费了九牛二虎之力终于把环境全配好之后,有一个问题摆在面前:
如何访问数据库?
通过继承HibernateDaoSupport我们有两个选择:
getSession().createQuery("from Users");
getHibernateTemplate().find( "FROM Users);
用哪个呢?困惑啊。
网上找了找资料都是推荐用getHibernateTemplate,原因说的不是很清楚。
于是我做了如下测试:
分别循环调用getSession().createQuery("from Users");getHibernateTemplate().find( "FROM Users);
1000次
结果getSession()很快就包无法建立连接了。而getHibernateTemplate屁事没有可以跑完。
通过后台观察,使用getSession会在数据库中留下很多SQL*Net message from client的连接,终止测试后连接自动释放。
而getHibernateTemplate则从头到尾都使用一个连接。
难道是getSession()不会自动释放连接?
于是我又分别循环调用getSession().createQuery("from Users");getHibernateTemplate().find( "FROM Users);
5次
发现当前端程序一结束,getSession的5个连接立刻就释放了。结合前面1000次时终止测试后连接自动释放,可以说明getSession()是会自动释放连接的。
结论:
1、getSession()和getHibernateTemplate都可以自动释放连接(当然你的配置要正确),但是在一个线程内getSession会get很多个session(就是开很多个会话、连接),很可能导致数据库连接超过上限。所以推荐使用getHibernateTemplate。
2、如果有些语句无法用getHibernateTemplate实现,可以使用getHibernateTemplate.execute使用HibernateCallback回调接口。
另:可以设定HibernateTemplate的AllowCreate为True,并在finally中关闭Session。也可以将true作为参数传递到super.getSession(..)方法中取得Session。这样也可以,就是麻烦点。
参见:
http://springframework.org/docs/api/org/springframework/orm/hibernate3/HibernateTemplate.html
http://www.mxjava.com/blog/article.asp?id=246
请各位高手不吝赐教。
另:用myEclipse自动生成的HibernateDAO代码中。4.1.1版本的myEclipse自动生成的findById方法使用的是getSession方法获得连接,不过在6.0中已经修改为使用getHibernateTemplate方法。5.0的没有测试。
转帖(二)http://jeoff.blog.51cto.com/186264/133434
* 使用 hql 语句进行操作
* @param hql HSQL 查询语句(使用回调函数访问外部变量,必须是final的)
* @param offset 开始取数据的下标
* @param length 读取数据记录数
* @return List 结果集
*/
public List getListForPage ( final String hql , final int offset , final int length ) {
public Object doInHibernate ( Session session ) throws HibernateException, SQLException {
Query query = session.createQuery ( hql ) ;
query.setFirstResult ( offset ) ;
query.setMaxResults ( length ) ;
List list = query.list ( ) ;
return list ;
}
}) ;
return list ;
}