Hibernate中的对象有三种状态:临时状态 (Transient),持久状态 (Persistent),游离状态(Detached)java
1. session三种状态:sql
1. 1.临时状态(Transient)数据库
由 new 命令开辟内存空间的 Java 对象,也就是平时所熟悉的普通 Java 对象,该对象未曾进行持久化,未与任何Session相关联,也叫自由态,只存在于内存中,而在数据库中没有相应数据。用new建立的对象,它没有持久化,没有处于Session中,处于此状态的对象叫临时对象;缓存
如: Student stu = new Student();session
临时对象特色:ide
(1) 不和 Session 实例关联性能
(2) 在数据库中没有和临时对象关联的记录ui
2. 2. 持久状态 (Persistent)spa
持久的实例在数据库中有对应的记录,并拥有一个持久化标识 (identifier).该对象与session关联而且在数据库中有相应数据。已经持久化,加入到了Session缓存中。如经过hibernate语句保存的对象。处于此状态的对象叫持久对象;hibernate
持久对象老是与 Session 和 Transaction 相关联,在一个 Session 中,对持久对象的改变不会立刻对数据库进行变动,而必须在Transaction 终止,也就是执行 commit() 以后,才在数据库中真正运行 SQL 进行变动,持久对象的状态才会与数据库进行同步。在同步以前的持久对象称为脏 (dirty) 对象。
临时对象转为持久对象:
(1) 经过 Session 的 save() 和 saveOrUpdate() 方法把一个临时对象与数据库相关联,这个临时对象就成为持久化对象。
(2) 使用 fine(),get(),load() 和 iterater() 待方法查询到的数据对象,将成为持久化对象。
持久化对象的特色:
(1) 和 Session 实例关联
(2) 在数据库中有和持久对象关联的记录
3. 3. 游离状态 (Detached)
与持久对象关联的 Session 被关闭后,对象就变为游离对象。对游离对象的引用依然有效,对象可继续被修改。
持久化对象脱离了Session的对象。如Session缓存被清空的对象。该对象已经持久化,但不在Session缓存中。处于此状态的对象叫游离对象;;
游离对象特色:
(1) 本质上和临时对象相同
(2) 只是临时对象多了一个数据库记录标识值 id.
持久对象转为游离对象:
当执行 close() 或 clear(),evict() 以后,持久对象会变为游离对象。
游离对象转为持久对象:
经过 Session 的 update(),saveOrUpdate() 和 lock() 等方法,把游离对象变为持久对象。
游离对象和临时对象异同:
二者都不会被Session关联,对象属性和数据库可能不一致;
游离对象有持久化对象关闭Session而转化而来,在内存中还有对象因此此时就变成游离状态了;
Hibernate和SQL的关系:
在操做了hibernate的方法如save()等后,并无直接生成sql语句,去操做数据库,而是把这些更新存入Session中,只有Session缓存要被更新时,底层的sql语句才能执行,数据存入数据库;
三种状态相互转化的状态图以下:
4 .结合 save(),update(),saveOrUpdate() 方法说明对象的状态
(1)Save() 方法将临时对象保存到数据库,对象的临时状态将变为持久化状态。当对象在持久化状态时,它一直位于 Session 的缓存中,对它的任何操做在事务提交时都将同步到数据库,所以,对一个已经持久的对象调用 save() 或 update() 方法是没有意义的。如:
Student stu = new Strudnet();
stu.setCarId(“200234567”);
stu.setId(“100”);
Session session = HibernateUtil.getSessionFactory().getCurrentSession();// 打开 Session
session.beginTransaction();// 开启事务
session.save(stu);
stu.setCardId(“20076548”);
session.save(stu); // 无效
session.update(stu); // 无效
session.getTransaction().commit();// 提交事务,
session.close();关闭 Session
(2)update() 方法两种用途从新关联游离对象为持久化状态对象,显示调用 update() 以更新对象。调用 update() 只为了关联一个游离对象到持久状态,当对象已是持久状态时,调用 update() 就没有多大意义了。如:
Session session = HibernateUtil.getSessionFactory().getCurrentSession();// 打开 Session
session.beginTransaction();// 开启事务
stu = (Student)session.get(Student.class,”123456”);
stu.setName(“Body”);
session.update(stu); // 因为 stu 是持久对象,必然位于 Session 缓冲中,对 stu 所作的变动将 // 被同步到数据库中。因此 update() 是没有意义的,能够不要这句效果同样的。
session.getTransaction().commit();// 提交事务,
session.close();关闭 Session
Hibernate 老是执行 update 语句,无论这个游离对象在离开 Session 以后有没有更改过,在清理缓存时 Hibernate 老是发送一条update 语句,以确保游离对象和数据库记录的数据一致。
如:
Student stu = new Strudnet();
stu.setCarId(“1234”);
Session session1 = HibernateUtil.getSessionFactory().getCurrentSession();// 打开 Session
session1.beginTransaction();// 开启事务
session1.save(stu);
session1.getTransaction().commit();// 提交事务,
session1.close();关闭 Session
stu.set(“4567”); // 对游离对象进行更改
Session session2 = HibernateUtil.getSessionFactory().getCurrentSession();// 打开 Session
session2.beginTransaction();// 开启事务
session2.update(stu);
session2.getTransaction().commit();// 提交事务,
session2.close();关闭 Session
注:即便把 session2.update(stu); 这句去掉,提交事务时仍然会执行一条 update() 语句。
若是但愿只有脱管对象改变了, Hibernate 才生成 update 语句,能够把映射文件中 <class> 标签的 select-before-update 设为true, 这种会先发送一条 select 语句取得数据库中的值,判断值是否相同,若是相同就不执行 update 语句。不过这种作法有必定的缺点,每次 update 语句以前老是要发送一条多余的 select 语句,影响性能。对于偶尔更改的类,设置才是有效的,对于常常要更改的类这样作是影响效率的。
(3)saveOrUpdate() 方法兼具 save() 和 update() 方法的功能,对于传入的对象, saveOrUpdate() 首先判断其是脱管对象仍是临时对象,而后调用合适的方法。
注:并不是到事务提交才对数据库操做,提交只是把变动持久化,实际上经过flush是能够操做的,只是这个变动只能在本事务看到而已。
一,Session.save(user)运行机理。
1,把User对象加入缓存中,使它变成持久化对象;
2,选用映射文件指定的标识生成ID;
3,在Session清理缓存时候执行:在底层生成一个insert sql语句,把对象存入数据库;
注意:在你执行Session.save(user)后,在Session清理缓存前,若是你修改user对象属性值,那么最终存入数据库的值将是最后修改的值;此过程当中ID不能被修改;
二,Session.delete(user)运行过程。
若是user是持久化对象,则执行删除操做,一样底层数据库的执行条件是:在Session清理缓存时候;
若是user是游离对象:
1,将user对象和Session关联,使之成为持久化对象;
2,而后按照user 是持久化对象的过程执行;
三态之间的转换方法:
①如何成为自由态?
对象经过构造方法成为自由态;持久态和游离态则经过session的delete方法成为自由态
②如何成为持久态?
对象能够由session的load或get方法直接成为持久态;自由态对象能够经过save,saveOrUpdate或persist方法成为持久态;游离态对象则能够经过update,saveOrUpdate成为持久态
③如何成为游离态?
游离态只能由持久态转换而来,经过close或clear方法实现。
即
临时态转换到持久态 save() saveOrUpdate()
持久态转换到临时态 delete()
持久态转换到游离态 evict() close() clear()
游离态转换到持久态 update() saveOrUpdate() lock()
几种转换方法的对比:
1.get 与load
都是从数据库中加载数据封装为java对象,使得java对象从自由态直接变为持久态;
可是有两点区别:①get返回对象能够为null,load返回值则始终不为null,找不到时会抛异常②get即时执行insert,而load则是在使用此对象时才执行insert
2.save,update与saveOrUpdate
save是将自由态转为持久态,而update是将游离态转为持久态,saveOrUpdate能够说是二者的综合,它执行时先判断对象的状态(主要是经过有无主键判断的),如果自由态,则save,如果游离态,则update
3.save与persist
二者都是将对象由自由态转为持久态,但返回值不一样:save返回主键值,而persist不返回
4,saveOrUpdate与merge
二者都是将自由态或游离态对象与数据库关联,但merge不改变对象的原有状态
此外,对clear与flush方法也做介绍。clear是将session中的对象所有变为游离态,是对象由持久态变为游离态的一种方法(另一种是关闭session);flush方法时为了使update操做能即时进行(正常状况下,只有在事务关闭时才进行update操做)。
Hibernate得到Session的两个方法
你们都知道,使用Hibernate对数据进行操做最重要的是得到一个Session。
首先得到一个SessionFactory,经过.config().buildSessionFactory(),得到一个SessionFactory 进而取得Session的方法在Hibernate里有两种:
1:Session session = sessionFactory.openSession();
该种方法是建立一个新的session,不论当前的环境中是否已经建立,都会建立;
且当session提交成功后,须要手动关闭这个session;
2:Session session = sessionFactory.getCurrentSession();
该方法是获取当前环境中的session,若环境中已经存在session,则取出当前的;若无session则建立一个新的,
只要当前的session提交后,该session会自动关闭,其后再使用getCurrentSession()方法,则是建立一个新的session;
使用以上两种方式还须要注意的是事务控制,若是使用第一种方法,openSession()也许会致使数据的不一致,
例如在用户管理功能模块中,咱们对用户的各类操做都须要作日志记录,此时要两张数据表同时提交成功,不然都不进行数据的写入;因此咱们要使用第二种方式来取得session。
特别强调:这两种取得Session的方法不能混用。缘由很简单,在Hibernate中,Session是一个接口。两个获取session的方法并不是是同一个实现,故不可混用!