Hibernate对象的三种状态(瞬时态,持久态,游离态)

2022-05-10 00:00:00 瞬时 三种 游离态

 1.  瞬时状态 (瞬态)

当我们通过Java的新的关键字来生成一个实体对象时,这时这个实体对象就处于自由状态,如下:
 客户客户=新客户(“zx”,27,图片);
这时客户对象就处于自由状态,为什么说客户对象处于自由状态呢?这是因为,此时的客户只是通过JVM获得了一块内存空间,还并没有通过会话对象的保存()方法保存进数据库,因此也就还没有纳入的Hibernate的缓存管理中,也就是说客户对象现在还自由的游荡于Hibernate的缓存管理之外。所以我们可以看出自由对象最大的特点就是,在数据库中不存在一条与它对应的记录。

瞬时对象特点:

(1) 不和 会话 实例关联

(2) 在数据库中没有和瞬时对象关联的记录

2.  持久状态 (持久性)

持久化对象就是已经被保存进数据库的实体对象,并且这个实体对象现在还处于休眠的缓存管理之中。这是对该实体对象的任何修改,都会在清理缓存时同步到数据库中。如下所示:
客户客户=新客户(“zx”,27,图片);
TX = session.beginTransaction();
session.save(客户);
客户=(客户)session.load(Customer.class,” 1” );
customer.setAge(28);
tx.commit();
这时我们并没有显示调用了Session.update()方法来保存更新,但是对实体对象的修改还是会同步更新到数据库中,因为此时客户对象通过保存方法保存进数据库后,已经是持久化对象了,然后通过负载方法再次加载它,它仍然是持久化对象,所以它还处于休眠缓存的管理之中,这时当执行tx.commit()方法时,休眠会自动清理缓存,并且自动将持久化对象的属性变化同步到到数据库中。

持久的实例在数据库中有对应的记录,并拥有一个持久化标识 (标识符)。

持久对象总是与 会话  交易 相关联,在一个 会议 中,对持久对象的改变不会马上对数据库进行变更,而必须在 交易 终止,也就是执行 的commit() 之后,才在数据库中真正运行 SQL  进行变更,持久对象的状态才会与数据库进行同步。在同步之前的持久对象称为脏 (脏) 对象。

瞬时对象转为持久对象:

(1) 通过 会话  节省()  saveOrUpdate()方法 方法把一个瞬时对象与数据库相关联,这个瞬时对象就成为持久化对象。

(2) 使用 细(),得到(),负载()  iterater() 待方法查询到的数据对象,将成为持久化对象。

持久化对象的特点:

(1)  会话 实例关联

(2) 在数据库中有和持久对象关联的记录

3.  脱管状态 (离体)

当一个持久化对象,脱离开的Hibernate的缓存管理后,它就处于游离状态,游离对象和自由对象的最大区别在于,游离对象在数据库中可能还存在一条与它对应的记录,只是现在这个游离对象脱离了休眠的缓存管理,而自由对象不会在数据库中出现与它对应的数据记录如下所示:
客户客户=新客户(“zx”,27,图片);
TX = session.beginTransaction();
session.save(客户);
客户=(客户)session.load(Customer.class,” 1” );
customer.setAge(28);
tx.commit();
session.close();
当会话关闭后,客户对象就不处于休眠的缓存管理之中了,但是此时在数据库中还存在一条与客户对象对应的数据记录,所以此时客户对象处于游离态

与持久对象关联的 会话 被关闭后,对象就变为脱管对象。对脱管对象的引用依然有效,对象可继续被修改。

脱管对象特点:

(1) 本质上和瞬时对象相同

(2) 只是比爱瞬时对象多了一个数据库记录标识值 的id。

持久对象转为脱管对象:

当执行 关闭()  清除(),逐出() 之后,持久对象会变为脱管对象。

瞬时对象转为持久对象:

通过 会话  更新(),saveOrUpdate()方法  锁() 等方法,把脱管对象变为持久对象。

三种状态相互转化的状态图如下:

《Hibernate对象的三种状态(瞬时态,持久态,游离态)》

4.  结合 save(),update(),saveOrUpdate() 方法说明对象的状态

(1)除() 方法将瞬时对象保存到数据库,对象的临时状态将变为持久化状态。当对象在持久化状态时,它一直位于 会话 的缓存中,对它的任何操作在事务提交时都将同步到数据库,因此,对一个已经持久的对象调用 保存() 更新() 方法是没有意义的如:

学生stu =新Strudnet();

stu.setCarId(“200234567”);

stu.setId(“100”);

//  打开 会议, 开启事务

session.save(STU);

stu.setCardId(“20076548”);

session.save(STU);  //  无效

了Session.update(STU); //  无效

//  提交事务,关闭 会话

(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  语句取得数据库中的值,判断值是否相同,如果相同就不执行 更新语句。不过这种做法有一定的缺点,每次 更新 语句之前总是要发送一条多余的 选择 语句,影响性能。对于偶尔更改的类,设置才是有效的,对于经常要更改的类这样做是影响效率的

(3)saveOrUpdate() 方法兼容 save()  update() 方法的功能,对于传入的对象, saveOrUpdate() 首先判断其是脱管对象还是临时对象,然后调用合适的方法。

    原文作者:飞鹰2016
    原文地址: https://blog.csdn.net/qq_16416231/article/details/79379762
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。

相关文章