做了一个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
相关推荐
1. **非线程安全**:由于`Session`不是线程安全的,因此在多线程环境中,每个线程都应有自己的`Session`实例,以避免数据存取的混乱。 2. **轻量级**:`Session`的创建和销毁相对快速,降低了资源的消耗。 3. **第一...
本示例将深入探讨Hibernate Session的生命周期及其使用,帮助你更好地理解和运用这个强大的工具。 Hibernate Session是Hibernate的核心接口,它是与数据库交互的主要接口。Session对象负责管理实体对象的状态,包括...
本文将深入探讨Hibernate中的核心概念——Session管理。 首先,理解Session在Hibernate中的角色至关重要。Session是Hibernate的主要工作单元,它是应用程序与数据库之间的桥梁。它负责保存、检索和更新Java对象,...
5. **线程安全**:`Session`不是线程安全的,因此在多线程环境中,包装类可能需要考虑如何正确管理和关闭`Session`实例,避免并发问题。 6. **简化API**:根据项目需求,包装类可以提供一些简洁的API,比如`...
Hibernate的Session接口是Java应用程序与Hibernate之间主要的运行时交互接口,它提供了对持久化对象的创建、读取和删除操作。Session的概念是基于对象的状态管理和数据库事务的,它的生命周期通常与一个物理事务绑定...
Hibernate Session的生命周期通常包括打开、使用和关闭三个阶段。在应用中,一个Session实例通常对应于一个数据库会话,用于执行数据库操作,如查询、保存、更新或删除实体。Session的创建通常通过SessionFactory...
openSession() 方法则不同,它不会将 Session 绑定到当前线程中,因此您需要手动关闭 Session。 getCurrentSession() 的配置 为了使用 getCurrentSession() 方法,您需要在 hibernate.cfg.xml 文件中添加以下配置...
3. **关闭Session**:在完成数据库操作后,记得关闭Session以释放资源。 通过以上介绍,我们可以看到Hibernate Session在数据库操作中的核心地位,正确理解和使用Session是掌握Hibernate的关键。实践中的具体应用...
如果一个持久化对象的Session被关闭或者对象从Session的缓存中移除,该对象就会变为游离状态。在这种状态下,对象仍然有ID,但与当前的Session失去了联系。如果对象的属性发生改变并希望更新到数据库,需要重新获得...
在设计系统时,可以考虑为每个请求分配一个 Session 实例,并在请求结束后及时关闭该 Session。 - **缓存机制**: Session 内部维护了一个缓存,用于存储当前工作单元中加载的所有对象。这个缓存通常被称为一级缓存,...
这些高级工具可以自动化处理事务管理和Session生命周期,减少出错的可能性。 在配置书籍库连接时,可能涉及以下内容: - **实体类**:为数据库表创建对应的Java类,使用Hibernate的注解(如@Entity、@Table、@Id等...
- 合理配置`hibernate.connection.release_mode`,确保在事务结束时正确关闭`Session`。 总结,Spring与Hibernate的集成使得我们可以在Spring的控制下更好地管理`Session`,实现更高效的事务处理,同时保持代码的...
通过begin()开启事务,Hibernate会自动创建Session,然后在commit()或rollback()时关闭Session。这种方法简化了代码,但可能导致Session生命周期过长,增加内存压力。 总的来说,理解和熟练运用Hibernate的缓存机制...
OSIV模式在Hibernate中,主要目的是在用户请求的整个生命周期内保持数据库会话,确保在视图层可以安全地访问到懒加载的对象,而不会引发已关闭的Session异常。 实现OSIV模式通常有以下步骤: 1. 配置拦截器:在...
- **脱管态(Detached)**:对象曾是持久态,但Session关闭后,对象与Session的关联断开,此时对象的状态可能已与数据库不同步。 **2. Hibernate的事务管理** 在Java应用中,事务处理是确保数据一致性的重要手段。...
然而,这也意味着如果Session在加载懒加载属性之前已经关闭,则会导致`HibernateException`(具体为`org.hibernate.LazyInitializationException`)。因此,正确地管理Session的生命周期对于实现Lazy Loading至关...
6. 关闭Session:每次操作完毕,记得关闭Session以释放资源。 总结起来,Hibernate3是Java开发中的强大工具,它极大地简化了数据库操作,提高了开发效率。理解并熟练掌握Hibernate3的核心概念和使用方法,对于任何...
在本教程中,我们将探讨如何利用Hibernate的注解功能,通过实体类自动生成数据库中的表。这极大地简化了数据库建模过程,同时也减少了手动编写SQL语句的工作量。 首先,我们需要理解Hibernate的核心概念。ORM框架如...
本文将深入探讨Hibernate中的事务处理,特别是“当前线程中的Session”这一概念,以及如何在实际开发中有效地利用它。 在Hibernate中,Session是与持久化层交互的主要接口,它负责对象的持久化、检索、更新和删除等...
7. **关闭资源**:最后记得关闭Session和SessionFactory。 ### 结论 通过以上介绍,我们了解到Hibernate是一个强大且灵活的ORM框架,它极大地简化了Java应用程序与数据库之间的交互过程。无论是对于初学者还是有...