hibernate 三种状态的转换
一、遇到的神奇的事情
使用jpa操作数据库,当我使用findall()方法查处一个list的对象后,给对这个list的实体进行了一些操作,并没有调用update 或者 saveorupdate方法,更改后的数据却神奇的保存到数据库里面去了。
最后简单粗暴的解决办法是把这份从数据里面查出来的list 复制了一份,然后再操作,再返回。数据就正常了,数据库也没更新。后面找了资料才发现是jpa是对hibernate的封装,底层是hibernate,这是hibernate的持久状态搞的鬼。
二、hibernate的三种状态
1. 瞬时状态 (transient)
当我们通过java的new关键字来生成一个实体对象时,这时这个实体对象就处于*状态,如下:
customer customer=new customer(“zx”,27,images);
这时customer对象就处于*状态,为什么说customer对象处于*状态呢?这是因为,此时customer只是通过jvm获得了一块内存空间,还并没有通过session对象的save()方法保存进数据库,因此也就还没有纳入hibernate的缓存管理中,也就是说customer对象现在还*的游荡于hibernate缓存管理之外。所以我们可以看出*对象最大的特点就是,在数据库中不存在一条与它对应的记录。
瞬时对象特点:
- 不和 session 实例关联
- 在数据库中没有和瞬时对象关联的记录
2. 持久状态 (persistent)
持久化对象就是已经被保存进数据库的实体对象,并且这个实体对象现在还处于hibernate的缓存管理之中。这是对该实体对象的任何修改,都会在清理缓存时同步到数据库中。如下所示:
customer customer=new customer(“zx”,27,images); tx=session.begintransaction(); session.save(customer); customer=(customer)session.load(customer.class,”1”); customer.setage(28); tx.commit();
这时我们并没有显示调用session.update()方法来保存更新,但是对实体对象的修改还是会同步更新到数据库中,因为此时customer对象通过save方法保存进数据库后,已经是持久化对象了,然后通过load方法再次加载它,它仍然是持久化对象,所以它还处于hibernate缓存的管理之中,这时当执行tx.commit()方法时,hibernate会自动清理缓存,并且自动将持久化对象的属性变化同步到到数据库中。
持久的实例在数据库中有对应的记录,并拥有一个持久化标识 (identifier).
持久对象总是与 session 和 transaction 相关联,在一个 session 中,对持久对象的改变不会马上对数据库进行变更,而必须在 transaction 终止,也就是执行 commit() 之后,才在数据库中真正运行 sql 进行变更,持久对象的状态才会与数据库进行同步。在同步之前的持久对象称为脏 (dirty) 对象。
瞬时对象转为持久对象:
- 通过 session 的 save() 和 saveorupdate() 方法把一个瞬时对象与数据库相关联,这个瞬时对象就成为持久化对象。
- 使用 fine(),get(),load() 和 iterater() 待方法查询到的数据对象,将成为持久化对象。
持久化对象的特点:
- 和 session 实例关联
- 在数据库中有和持久对象关联的记录
3. 脱管状态 (detached)
当一个持久化对象,脱离开hibernate的缓存管理后,它就处于游离状态,游离对象和*对象的最大区别在于,游离对象在数据库中可能还存在一条与它对应的记录,只是现在这个游离对象脱离了hibernate的缓存管理,而*对象不会在数据库中出现与它对应的数据记录。如下所示:
customer customer=new customer(“zx”,27,images); tx=session.begintransaction(); session.save(customer); customer=(customer)session.load(customer.class,”1”); customer.setage(28); tx.commit(); session.close();
当session关闭后,customer对象就不处于hibernate的缓存管理之中了,但是此时在数据库中还存在一条与customer对象对应的数据记录,所以此时customer对象处于游离态
与持久对象关联的 session 被关闭后,对象就变为脱管对象。对脱管对象的引用依然有效,对象可继续被修改。
脱管对象特点:
- 本质上和瞬时对象相同
- 只是比爱瞬时对象多了一个数据库记录标识值 id.
持久对象转为脱管对象:
当执行 close() 或 clear(),evict() 之后,持久对象会变为脱管对象。
瞬时对象转为持久对象:
通过 session 的 update(),saveorupdate() 和 lock() 等方法,把脱管对象变为持久对象。
三、三种状态的转换
四、举例子
1、结合 save(),update(),saveorupdate() 方法说明对象的状态
(1)save() 方法将瞬时对象保存到数据库,对象的临时状态将变为持久化状态。当对象在持久化状态时,它一直位于 session 的缓存中,对它的任何操作在事务提交时都将同步到数据库,因此,对一个已经持久的对象调用 save()或 update() 方法是没有意义的。如:
student stu = new strudnet(); stu.setcarid(“200234567”); stu.setid(“100”); // 打开 session, 开启事务 session.save(stu); stu.setcardid(“20076548”); session.save(stu); // 无效 session.update(stu); // 无效 // 提交事务,关闭 session
(2)update() 方法两种用途重新关联脱管对象为持久化状态对象,显示调用 update() 以更新对象。调用 update() 只为了关联一个脱管对象到持久状态,当对象已经是持久状态时,调用 update() 就没有多大意义了。如:
// 打开 session ,开启事务 stu = (student)session.get(student.class,”123456”); stu.setname(“body”); session.update(stu); // 由于 stu 是持久对象,必然位于 session 缓冲中, 对 stu 所做的变更将 // 被同步到数据库中。所以 update() 是没有意义的,可以不要这句效果一样的。 // 提交事务,关闭 session hibernate 总是执行 update 语句,不管这个脱管对象在离开 session 之后有没有更改过,在清理缓存时 hibernate总是发送一条 update 语句,以确保脱管对象和数据库记录的数据一致,如: student stu = new strudnet(); stu.setcarid(“1234”); // 打开 session1, 开启事务 session1.save(stu); // 提交事务,关闭 session1 stu.set(“4567”); // 对脱管对象进行更改 // 打开 session2, 开启事务 session2.update(stu); // 提交事务,关闭 session2
注:即使把 session2.update(stu); 这句去掉,提交事务时仍然会执行一条 update() 语句。
如果希望只有脱管对象改变了, hibernate 才生成 update 语句,可以把映射文件中 <class> 标签的 select-before-update 设为 true, 这种会先发送一条 select 语句取得数据库中的值,判断值是否相同,如果相同就不执行 update语句。不过这种做法有一定的缺点,每次 update 语句之前总是要发送一条多余的 select 语句,影响性能。对于偶尔更改的类,设置才是有效的,对于经常要更改的类这样做是影响效率的。
(3)saveorupdate() 方法兼具 save() 和 update() 方法的功能,对于传入的对象, saveorupdate() 首先判断其是脱管对象还是临时对象,然后调用合适的方法。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持!