做了一个SSH集成作为开发环境,Service层只是注入了sessionFactory,没有配置事务管理。
发现进行几次操作后就会当掉,但是Tomcat服务没有中断。
我首先鉴定是连接没有释放,可能是连接池太小吧,我设置大了仍然不管用。
我想着可能是因为连接是需要手动释放的,于是进行手动关闭来进行验证:
@Override
public int getResourceListCount() {
int count = 0;
Session session = this.getSession();
String hql = "select count(*) from Resource u order by u.id desc";
Query query = session.createQuery(hql);
try {
count = Integer.parseInt(query.uniqueResult().toString());
} catch (Exception e) {
}
session.close();
return count;
}
将会话手动进行关闭,发现就没有问题了。
网上查了一下才知道,Hibernate的会话不会自动关闭,也就是sessionFactory每次创建一个session,如果你没有手动关闭,那么就会连接不释放!
毕竟是做集成,想到还没有配置Spring的事务管理,于是乎马上配置了一下,我采用的是代理方式:
<!-- 数据源 --> <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close"> <property name="driverClassName" value="com.mysql.jdbc.Driver" /> <property name="url" value="jdbc:mysql://192.168.0.244:3306/test?useUnicode=true&characterEncoding=UTF-8" /> <property name="username" value="root" /> <property name="password" value="root" /> <!-- 连接池启动时的初始值 --> <property name="initialSize" value="10" /> <!-- 连接池的最大值 --> <property name="maxActive" value="10" /> <!-- 最大空闲值.当经过一个高峰时间后,连接池可以慢慢将已经用不到的连接慢慢释放一部分,一直减少到maxIdle为止 --> <property name="maxIdle" value="20" /> <!-- 最小空闲值.当空闲的连接数少于阀值时,连接池就会预申请去一些连接,以免洪峰来时来不及申请 --> <property name="minIdle" value="10" /> <property name="defaultAutoCommit" value="true" /> </bean> <!-- 会话工厂 --> <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"> <property name="dataSource" ref="dataSource" /> <property name="mappingLocations"> <list> <value>classpath:/com/nms/entity/**/*.hbm.xml</value> </list> </property> <property name="hibernateProperties"> <props> <prop key="hibernate.dialect"> org.hibernate.dialect.MySQL5Dialect </prop> <prop key="hibernate.show_sql">true</prop> <prop key="hibernate.format_sql">true</prop> </props> </property> </bean> <!-- 事物管理器 --> <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager"> <property name="sessionFactory" ref="sessionFactory"></property> </bean> <!-- 声明事物 --> <bean id="base" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean" abstract="true"> <property name="transactionManager" ref="transactionManager"> </property> <property name="transactionAttributes"> <props> <prop key="*">PROPAGATION_REQUIRED</prop> </props> </property> </bean>
事务均采用PROPAGATION_REQUIRED
然后每个服务类都配置一个代理对象:
<bean id="userDao" class="com.nms.service.common.impl.UserDaoImpl"> <property name="sessionFactory" ref="sessionFactory" /> </bean> <!-- 代理类 --> <bean id="userDaoAgency" parent="base"> <property name="target" ref="userDao"></property> </bean>
在配置Struts2的Action时需要服务类都是用代理对象:
<bean id="userAction" class="com.nms.action.common.UserAction" scope="prototype"> <property name="userDao" ref="userDaoAgency"></property> <property name="jdbcRoleDao" ref="jdbcRoleDao"></property> </bean>
这样,每次操作都在不同的事务中,一个事务结束就会由Spring来处理这个已经结束的会话。
但是还有一个问题,在服务层你的会话结束了,但是在前台可能使用是一个Hibernate代理对象,就是说没有实际去查询数据,这就需要你的对象不能进行懒加载。
<class name="User" table="common_user" lazy="false">
查询时在服务层直接把对象查询出来,不然你到JSP页面去取值时会报错
org.hibernate.LazyInitializationException: could not initialize proxy - no Session
当然我这只是简单配置下做个测试,对于Spring的事务配置那是有很多种方式的,回头我再总结一下给大家分享!
请您到ITEYE看我的原创:http://cuisuqiang.iteye.com
或支持我的个人博客,地址:http://www.javacui.com