- 浏览: 3008815 次
- 性别:
- 来自: 河南
文章分类
- 全部博客 (340)
- Java综合 (26)
- 程序人生 (53)
- RIA-ExtJS专栏 (18)
- RIA-mxGraph专栏 (4)
- RIA-Flex4专栏 (43)
- 框架-Spring专栏 (16)
- 框架-持久化专栏 (22)
- 框架-Struts2专栏 (11)
- 框架-Struts专栏 (12)
- SQL/NOSQL (12)
- 报表/图表 (2)
- 工作流 (5)
- XML专栏 (4)
- 日常报错解决方案 (5)
- Web前端-综合 (12)
- Web/JSP (14)
- Web前端-ajax专栏 (14)
- Web前端-JQuery专栏 (9)
- IDE技巧 (6)
- FILE/IO (14)
- 远程服务调用 (2)
- SSO单点登录 (2)
- 资源分享 (22)
- 云计算 (1)
- 项目管理 (3)
- php专栏 (1)
- Python专栏 (2)
- Linux (1)
- 缓存系统 (1)
- 队列服务器 (1)
- 网络编程 (0)
- Node.js (1)
最新评论
-
hui1989106a:
我的也不能解压,360和好压都试了,都不行
《Spring in Action》完整中文版分享下载 -
temotemo:
这些example有些过时了,官方建议使用HBase-1.0 ...
Java操作Hbase进行建表、删表以及对数据进行增删改查,条件查询 -
zy8102:
非常感谢~
HeadFirst系列之七:《深入浅出SQL》原版高清PDF电子书分享下载 -
zy8102:
重命名了一下搞定了
HeadFirst系列之七:《深入浅出SQL》原版高清PDF电子书分享下载 -
zy8102:
为什么下载以后老解压不了呢?
HeadFirst系列之七:《深入浅出SQL》原版高清PDF电子书分享下载
说到Hibernate那么最核心的就是它的有关数据库的增删改查操作了,而这些操作中增删改无不依赖着一个关键的对象session,那么提到session就不得不提session中对象的四个状态
看图便知道,通常情况下,大家都认为session中的对象存在三种状态:瞬时(transitent)、持久化(persistent)以及
托管(detached)。不过有时还存一种观点,认为应该是四种状态,即还存在一种移除(removed)状态。对于这两种观点呢我们暂不追究到底以哪个为依据,因为到现在还没统一的定论
本篇文章中,为了全面讲解,所以移除状态我也也涉及到
Session中对象的状态
1) 瞬时状态(transient): 新创建的对象。没有和某个Session进行关联。没有对象标识符(OID)。
2) 持久化状态(persistent): 与某个session进行关联。有对象标识符。数据库表中有对应的记录。
session在清理缓存时,会把此对象的数据与数据库表的数据进行同步。
3) 脱管状态(detached): 脱离了Session的管理。有对象标识符。数据库表中有对应的记录。
不保证此对象的数据与数据库表的数据是否同步。
4) 移除状态(removed): 与某个session进行关联。有对象标识符,数据库表中有对应的记录。
session在清理缓存时,会把数据库表对应的记录删除掉。这个对象不能再去使用它.
针对这些解释呢,基于第一篇文章中的类和配置文件,这里只提供一个测试类,方法上都有更详细的解释
package com.javacrazyer.test;
import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.junit.BeforeClass;
import org.junit.Test;
import com.javacrazyer.common.HibernateUtil;
import com.javacrazyer.domain.Student;
/**
* 使用Hibernate API完成CRUD 更复杂的持久化操作需要使用到Query接口
*
*/
public class HibernateTest {
private static SessionFactory factory;
@BeforeClass
public static void init() {
factory = HibernateUtil.getSessionFactory();
}
@Test
public void testSessionCache() {
Session session = factory.openSession();
Transaction tx = session.beginTransaction();
Student stu = (Student) session.get(Student.class, 1);
System.out.println(stu);
Student stu2 = (Student) session.get(Student.class, 1);
System.out.println(stu2);
stu2.setName("xkk");
System.out.println(stu2);
/*flush()将数据库与缓存中的数据同步,不是必须调用的*/
session.flush(); // 手动清理缓存
/*
* clear()写在flush后面,执行后才会引起缓存数据变化,session.flush()的调用牵扯到事务,
* 首先我们知道在执行事务之前都会将AutoCommit设置为false【手动提交方式,因为默认是true
* 自动提交的】 当AutoCommit 为false时我们执行完事务就要调用到session.flush();
* session.clear();一切处理完后我们要close掉当前的这个session
*/
session.clear();
stu2.setScore(77.00);
System.out.println(stu2);
tx.commit();
session.close();
}
@Test
public void testObjectStatus() {
Student stu = new Student(); // transient瞬时状态
stu.setName("ww");
stu.setGender(false);
stu.setAge(38);
stu.setScore(65.5);
Session session = factory.openSession();
Transaction tx = session.beginTransaction();
session.save(stu); // persistent持久化状态
System.out.println(stu);
tx.commit();
session.close();
System.out.println(stu); // detached 脱管
Session session2 = factory.openSession();
session2.beginTransaction();
session2.save(stu);
System.out.println(stu);// 脱管状态--> 持久化状态 (不建议用save方法来操作脱管对象)
session2.getTransaction().commit();
session2.close();
}
/*
* get()方法:先查找session缓存中是否已经存在此标识符指定的对象,如果存在,直接使用.
* 否则发出SQL语句从数据库中获取.如果数据库中也不存在,返回null
*/
@Test
public void testGet() {
Session session = factory.openSession();
session.beginTransaction();
Student stu = (Student) session.get(Student.class, 1);
System.out.println(stu);
session.getTransaction().commit();
session.close();
}
/*
* load()方法先查找session缓存中是否已经存在此标识符指定的对象,如果存在,直接使用.
* 否则Hibernate会为此标识符对象产生一个代理对象,实现延迟加载(懒加载)的功能。这个代理对象包含有OID。
* 当要使用到此对象的非OID属性值,才发出SQL语句去数据库中获取。
* 如果数据库中也不存在,返回InvocationTargetException异常。
*/
@Test
public void testLoad() {
Session session = factory.openSession();
session.beginTransaction();
Student stu = (Student) session.load(Student.class, 5);
System.out.println(stu.getId());
System.out.println(stu);
session.getTransaction().commit();
session.close();
}
/*
* delete()方法:持久化状态 --> 移除状态 .
* 注意:处理移除状态的对象不要再去使用它,因为,在session清理缓存时,数据库表中对应的数据会被删除掉.
*/
@Test
public void testDelete() {
Session session = factory.openSession();
session.beginTransaction();
Student stu = (Student) session.load(Student.class, 7);
session.delete(stu);
System.out.println(stu);
session.getTransaction().commit();
session.close();
System.out.println(stu);
}
@Test
/* update()方法: 重附被修改的脱管对象,成为持久化对象 */
public void testUpdate() {
Session session = factory.openSession();
session.beginTransaction();
Student stu = (Student) session.load(Student.class, 8);
stu.setName("更新持久化状态的对象");
session.getTransaction().commit();
session.close();
System.out.println(stu);
stu.setName("修改脱管对象");
Session session2 = factory.openSession();
session2.beginTransaction();
session2.update(stu);
session2.getTransaction().commit();
session2.close();
}
@Test
/*
* saveOrUpdate()方法:
* 1) 瞬时对象,执行类似save()的功能
* 2) 脱管对象,如果在当前session缓存中不存在同OID的对象,就执行类似update()的功能。否则,抛出异常。
*/
public void testSaveOrUpdate() {
Session session = factory.openSession();
session.beginTransaction();
Student stu = (Student) session.get(Student.class, 9);
System.out.println(stu);
session.getTransaction().commit();
session.close();
// 处理脱管状态
stu.setName("9哥");
Session session2 = factory.openSession();
session2.beginTransaction();
session2.saveOrUpdate(stu);
System.out.println(stu);
session2.getTransaction().commit();
session2.close();
System.out.println(stu);
}
@Test
public void testSaveOrUpdate2() {
Student stu = new Student();
stu.setId(19); // 对象没有与session关联,并且OID有值,就被认为是脱管对象
Session session = factory.openSession();
session.beginTransaction();
session.saveOrUpdate(stu);
session.getTransaction().commit();
session.close();
}
@Test
public void testSaveOrUpdate3() {
Student stu = new Student();
stu.setId(9); // 对象没有与session关联,并且OID有值,就被认为是脱管对象
stu.setName("su");
Session session = factory.openSession();
session.beginTransaction();
session.get(Student.class, 9);
session.saveOrUpdate(stu);
session.getTransaction().commit();
session.close();
}
@Test
/*
* merge()方法:
* 1) 瞬时对象,执行类似save()的功能
* 2)脱管对象:如果在当前session缓存中不存在同OID的对象,就执行类似update()的功能。
* 否则,把传入的对象数据合并到缓存中的对象,返回缓存中的对象。
* 3) 经常用来替代update()和saveOrUpdate()方法。
*/
public void testmerge() {
Student stu = new Student();
stu.setId(9); // 对象没有与session关联,并且OID有值,就被认为是脱管对象
stu.setName("su");
Session session = factory.openSession();
session.beginTransaction();
session.get(Student.class, 9);
stu = (Student) session.merge(stu);
session.getTransaction().commit();
session.close();
}
}
补充说明:sessionFactory.getCurrentSession()和sessionFactory.openSesion()的区别介绍
1. 如果使用的是getCurrentSession来创建session的话,在commit后,session就自动被关闭了,也就是不用再session.close()了。但是如果使用的是openSession方法创建的session的话,那么必须显示的关闭session,也就是调用session.close()方法。这样commit后,session并没有关闭
2.getCurrentSession创建的session会和绑定到当前线程,而openSession不会。getCurrentSession创建的线程会在事务回滚或事物提交后自动关闭,而openSession必须手动关闭,
3. 使用SessionFactory.getCurrentSession()需要在hibernate.cfg.xml中如下配置:
* 如果采用jdbc独立引用程(本地事务:JDBC事务)序配置如下:
<property name="hibernate.current_session_context_class">thread</property>
* 如果采用了JTA事务配置(全局事务:JTA事务)如下
<property name="hibernate.current_session_context_class">jta</property>
4.getCurrentSession () 使用当前的session
openSession() 重新建立一个新的session
总结:
getCurrentSession和openSession无论是那种方式,如果是纯JDBC项目的话,那你必须手动写上事务的开启和提交,openSession事务提交后还得手动写session.close()关闭,尽管是这样也不一定真的关了;getCurrentSession提交事务后会自动关闭session所以不用手动写session.close()
其实际项目(我指的是SSH项目)中由于在Spring中配置有事务管理,所以我们用getCurrentSession时手动写的关于事务的代码配置都不用写了
//openSession()方法的测试,必须在事务提交后关闭session
@Test
public void openSessionDelete(){
Session session = factory.openSession();
Transaction tx = session.beginTransaction();
Student stu = (Student) session.get(Student.class, 1);
session.delete(stu);
tx.commit();
session.close();
}
//getCurrentSession()方法,不需要关闭session
@Test
public void getCurrentSessionDelete(){
Session session = factory.getCurrentSession();
Transaction tx = session.beginTransaction();
Student stu = (Student) session.get(Student.class,2);
session.delete(stu);
tx.commit();
}
相比之下getCurrentSession()还是最适合的
这句红色标记的话我要用下面的话来解释下:
在一个应用程序中,如果DAO 层使用Spring 的hibernate 模板,通过Spring 来控制session 的生命周期,则首选getCurrentSession ()。
使用Hibernate的大多数应用程序需要某种形式的“上下文相关的” session,特定的session在整个特定的上下文范围内始终有效。然而,对不同类型的应用程序而言,要为什么是组成这种“上下文”下一个定义通常是困难的;不同的上下文对“当前”这个概念定义了不同的范围。在3.0版本之前,使用Hibernate的程序要么采用自行编写的基于 ThreadLocal的上下文session,要么采用HibernateUtil这样的辅助类,要么采用第三方框架(比如Spring或Pico),它们提供了基于代理(proxy)或者基于拦截器(interception)的上下文相关session。
从3.0.1版本开始,Hibernate增加了SessionFactory.getCurrentSession()方法。一开始,它假定了采用JTA事务,JTA事务定义了当前session的范围和上下文(scope and context)。Hibernate开发团队坚信,因为有好几个独立的JTA TransactionManager实现稳定可用,不论是否被部署到一个J2EE容器中,大多数(假若不是所有的)应用程序都应该采用JTA事务管理。基于这一点,采用JTA的上下文相关session可以满足一切需要。
在 SessionFactory 启动的时候, Hibernate 会根据配置创建相应的 CurrentSessionContext ,在 getCurrentSession() 被调用的时候,实际被执行的方法是 CurrentSessionContext.currentSession() 。在 currentSession() 执行时,如果当前 Session 为空, currentSession 会调用 SessionFactory 的 openSession 。所以 getCurrentSession() 对于 Java EE 来说是更好的获取Session 的方法。
那么跟跟openSession相比,getCurrentSession在使用上有什么注意的呢? 到现在发现的一个就是,由于getCurrentSession方法返回的session在做事务的commit时, session可能会自动给关掉,这样若自己的代码中再调用session.close时就抛出了"Session was already closed"异常。
在HB3.X 版本中提供了一个getCurrentSession() 这个方法,这个方法和早期使用的openSession() 是有区别的。
openSession() ,表示创建了一个新的session 对象,当你使用完了以后就要必须调用close 方法来关闭当前的session 。getCurrentSession() ,总是会返回“ 当前的” 工作单元。Session 在第一次被使用的时候,即第一次调用getCurrentSession() 的时候,其生命周期就开始。然后她被Hibernate 绑定到当前线程。当事物结束的时候,不管是提交还是回滚,Hibernate 会自动把Session 从当前线程剥离,并且关闭。若在次调用getCurrentSession() ,会得到一个新的Session, 并且开始一个新的工作单元。这是Hibernate 最广泛的thread-bound model ,支持代码灵活分层( 事物划分和数据访问代码的分离)
评论
Configuration config = new AnnotationConfiguration().configure(); SessionFactory sessionFactory = config.buildSessionFactory(); for(int i=0;i<500;i++){ Session session = sessionFactory.openSession(); session.createCriteria(SysConfig.class).list(); //session.close(); }
这里我开了500个session,每个都做了一次数据库的查询,而且都没有执行关闭,为何不报数据库的连接异常,我把连接数提升到5000照样正常执行,难道通过openSession()开启的数据库连接也可能自己释放吗?
又 如果这个方法操作结束后,session对象被JVM回收了,连接会不会自动被释放呢?
http://505629625-qq-com.iteye.com/admin/blogs/804385
发表评论
-
ibatis常用16条SQL语句
2011-07-29 11:08 26091(1) 输入参数为单个值 ... -
iBATIS与Hibernate的异同
2010-12-30 14:47 3713Hibernate Hibernate是一个开放源代 ... -
iBATIS与Spring整合
2010-12-30 14:34 4881接着iBATIS的入门实例来说,ibatis与Spring的 ... -
Hibernate温习(17)--OpenSessionInView模式
2010-11-11 17:04 4564在WEB应用程序中,视图(JSP或Servlet)可能会通过导 ... -
Hibernate温习(16)--性能优化之缓存管理
2010-11-11 16:02 17811.缓存概述 缓存(cache) ... -
Hibernate温习(15)--性能优化之抓取策略
2010-11-11 14:51 3522抓取策略(fetching strategy) 是指:当应 ... -
Hibernate温习(14)--性能优化之延迟加载机制
2010-11-11 11:33 1776延迟加载 延迟加载 ... -
Hibernate温习(13)--Hibernate程序性能优化的考虑要点
2010-11-11 10:55 1341本文依照HIBERNATE帮助文档,一些网络书籍及项目经 ... -
Hibernate温习(12)--基于注解方式的各种映射全面总结
2010-11-11 10:43 36001. 使用Hibernate Annotation来做对 ... -
Hibernate温习(11)--多事务并发访问控制
2010-11-11 10:25 7437在并发环境,一个数据库系统会同时为各种各样的客户程序 ... -
Hibernate温习(10)--应用程序中的事务管理
2010-11-11 09:26 1939事务的定义 事务就是指作为单个逻辑工作单元执行的一组数据操作 ... -
Hibernate温习(9)--有关Hibernate升级后注解方式的对象关系映射
2010-11-10 17:06 3341我要说的升级指的是我实际中遇到的,由于我之前的项目中Hiber ... -
Hibernate进行测试时时常会有的错误
2010-11-10 14:31 1502在hibernate程序中,如果使用JUNIT进行测试的话,第 ... -
Hibernate温习(8)--使用JPA
2010-11-10 11:00 2382这次讲的JPA前一篇文章都有所介绍,这里呢就是结合hibern ... -
Hibernate温习(7)--JPA回顾
2010-11-10 10:39 1978什么是JPA JPA(Java Pers ... -
Hibernate温习(6)--单向一对一外键关联映射
2010-11-05 10:24 2617hibernate一对一唯一外键关联映射(单向关联Citize ... -
Hibernate温习(5)--集合属性映射
2010-11-04 22:37 2451Hibernate的集合属性映射 ... -
Hibernate温习(1)--最基础的示例
2010-11-04 14:22 3453有关Spring的知识大部分都已经温习完毕,今天开始转向Hib ... -
ibatis入门实例讲解
2010-10-25 10:39 12993之前大家上网的ibatis官方网站:http://www.i ... -
HIbernate温习(2)--连接池配置总结基于第三方c3p0和proxool
2010-07-26 14:42 3443一直都在用连接池技术,也是个好习惯,但用连接 ...
相关推荐
《深入理解Hibernate配置与映射:hibernate-configuration-3.0.dtd与hibernate-mapping-3.0.dtd解析》 在Java世界里,Hibernate作为一款强大的对象关系映射(ORM)框架,极大地简化了数据库操作。而`hibernate-...
1. **源代码**:包括了Hibernate的核心模块和其他相关模块的源代码,这对于深入理解Hibernate的工作原理和进行二次开发非常有帮助。 2. **库文件**:包含jar包,如hibernate-core.jar、hibernate-entitymanager.jar...
hibernate-jpa-2.0-api-1.0.1.Final.jar
hibernate-jpa-2.0-api-1.0.1.Final-sources.jar hibernate jpa 源代码
3. **事务管理**:使用Session的`beginTransaction()`、`commit()`和`rollback()`方法进行事务控制。 4. **查询数据**:使用HQL或Criteria API进行复杂的数据检索。 六、性能优化 - 使用二级缓存提高性能,如...
hibernate-commons-annotations-4.0.1.Final.jar
在这个`hibernate-release-5.0.7.Final`版本中,包含了所有相关的jar包,为开发者提供了一个完整的Hibernate ORM解决方案。 在Java开发中,jar(Java Archive)包是Java类库的打包形式,它包含了一系列的类文件和...
总结,Hibernate 5.0.0.Final版本为Java开发者提供了更为强大的ORM功能,通过深入理解和熟练运用,我们可以构建更高效、更健壮的Java应用。对于希望提升数据库操作效率和简化数据管理的开发者来说,这是一个值得深入...
《深入理解Hibernate-CGLIB-2.1_3在Java开发中的应用》 在Java开发领域,Hibernate作为一款强大的对象关系映射(ORM)框架,极大地简化了数据库操作。而CGLIB则是一个代码生成库,它允许开发者在运行时动态创建子类...
因为Hibernate在读出hbm.xml文件时需要通过网络读取到hibernate-mapping-3.0.dtd 文件。 如果没有网络不能正常工作。 所以提供上述文件。 以及hibernate-mapping-3.0.dtd,hibernate-configuration-3.0.dtd提供下载...
3. Hibernate 5.0.7.Final新特性 - 支持JSR-349 Bean Validation 1.1:增强了实体验证功能,允许在对象层面进行数据验证。 - Entity Manager改进:提供了更多的灵活性,如多数据源支持和更丰富的查询选项。 - ...
hibernate-mapping-3.0.dtd 最新从官网下载,无损无修改
【描述】"这个压缩文件包含的是Hibernate框架4.2.4.Final版本的相关组件,是开发人员在使用Hibernate进行Java应用程序开发时的重要资源。Hibernate是一个开源的对象关系映射(ORM)框架,它简化了数据库操作,使得...
例如,通过配置`hibernate.cfg.xml`文件设置数据库连接,编写实体类并使用注解进行ORM映射,然后使用SessionFactory创建Session对象来执行CRUD操作。同时,Hibernate提供了HQL(Hibernate Query Language)和...
2. 数据库连接与会话:学习如何配置Hibernate,创建SessionFactory,以及如何打开和关闭Session。 3. CRUD操作:掌握如何通过Hibernate进行增删改查的基本操作。 4. 查询语言:理解HQL(Hibernate Query Language)...
hibernate-entitymanager-3.4.0.GA.rar hibernate-entitymanager-3.4.0.GA.rar hibernate-entitymanager-3.4.0.GA.rar hibernate-entitymanager-3.4.0.GA.rar hibernate-entitymanager-3.4.0.GA.rar hibernate-...
总结,Hibernate ORM 5.3.2.Final版本是Java开发者实现高效数据库操作的强大工具,其丰富的特性和优化,使得开发者能更加专注于业务逻辑,而非底层数据库操作。正确理解和运用Hibernate,将极大地提升开发效率和软件...
这个jar包是使用Hibernate进行开发时必不可少的部分,它定义了Hibernate API,提供了Entity Manager和Session接口,以及对事务处理的支持。 其次,`javassist-3.15.0-GA.jar`是Java编程辅助工具包,它允许在运行时...
6. **hibernate-jpa-2.1-api-1.0.0.Final.jar**:提供了JPA 2.1的API规范,使得Hibernate能够遵循这个标准,提供兼容性并支持JPA的相关特性。 7. **hibernate-commons-annotations-5.0.1.Final.jar**:包含了一些...
3. Query与Criteria:提供了两种查询方式,HQL(Hibernate Query Language)是一种面向对象的查询语言,类似于SQL;Criteria API则提供了更动态、更面向对象的查询方式。 三、实体管理 Hibernate通过@Entity注解...