Hibernate 是一个开放源代码的对象关系映射框架,它对 JDBC 进行了非常轻量级的对象封装,使得 Java 程序员可以随心所欲的使用对象编程思维来操纵数据库。 Hibernate 可以应用在任何使用 JDBC 的场合,既可以在 Java 的客户端程序使用,也可以在 Servlet/JSP 的 web 应用中使用,最具革命意义的是,Hibernate 可以在应用 EJB 的 J2EE 架构中取代 CMP,完成数据持久化的重任。
本文主要介绍了 hibernate 三种状态的转换。具有很好的参考价值。下面跟着小编一起来看下吧
一、遇到的神奇的事情使用 jpa 操作数据库,当我使用 findAll()方法查处一个 List 的对象后,给对这个 list 的实体进行了一些操作,并没有调用 update 或者 saveOrUpdate 方法,更改后的数据却神奇的保存到数据库里面去了。
最后简单粗暴的解决办法是把这份从数据里面查出来的 List 复制了一份,然后再操作,再返回。数据就正常了,数据库也没更新。后面找了资料才发现是 jpa 是对 hibernate 的封装,底层是 hibernate,这是 hibernate 的持久状态搞的鬼。
二、hibernate 的三种状态1. 瞬时状态 (Transient)当我们通过 Java 的 new 关键字来生成一个实体对象时,这时这个实体对象就处于自由状态,如下:
- Customer customer =newCustomer("zx",27, images);
这时 customer 对象就处于自由状态,为什么说 customer 对象处于自由状态呢?这是因为,此时 customer 只是通过 JVM 获得了一块内存空间,还并没有通过 Session 对象的 save() 方法保存进数据库,因此也就还没有纳入 Hibernate 的缓存管理中,也就是说 customer 对象现在还自由的游荡于 Hibernate 缓存管理之外。所以我们可以看出自由对象最大的特点就是,在数据库中不存在一条与它对应的记录。
瞬时对象特点:持久化对象就是已经被保存进数据库的实体对象,并且这个实体对象现在还处于 Hibernate 的缓存管理之中。这是对该实体对象的任何修改,都会在清理缓存时同步到数据库中。如下所示:
- Customer customer =newCustomer("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) 对象。
瞬时对象转为持久对象:当一个持久化对象,脱离开 Hibernate 的缓存管理后,它就处于游离状态,游离对象和自由对象的最大区别在于,游离对象在数据库中可能还存在一条与它对应的记录,只是现在这个游离对象脱离了 Hibernate 的缓存管理,而自由对象不会在数据库中出现与它对应的数据记录。如下所示:
- Customer customer =newCustomer("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 被关闭后,对象就变为脱管对象。对脱管对象的引用依然有效,对象可继续被修改。
脱管对象特点:当执行 close() 或 clear(),evict() 之后,持久对象会变为脱管对象。
瞬时对象转为持久对象:通过 Session 的 update(),saveOrUpdate() 和 lock() 等方法,把脱管对象变为持久对象。
三、三种状态的转换四、举例子 1、结合 save(),update(),saveOrUpdate() 方法说明对象的状态(1)Save() 方法将瞬时对象保存到数据库,对象的临时状态将变为持久化状态。当对象在持久化状态时,它一直位于 Session 的缓存中,对它的任何操作在事务提交时都将同步到数据库,因此,对一个已经持久的对象调用 save() 或 update() 方法是没有意义的。如:
- Student stu =newStrudnet();
- 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() 就没有多大意义了。如:
注:即使把 session2.update(stu); 这句去掉,提交事务时仍然会执行一条 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 =newStrudnet();
- stu.setCarId("1234");
- // 打开 Session1, 开启事务
- session1.save(stu);
- // 提交事务,关闭 Session1
- stu.set("4567");// 对脱管对象进行更改
- // 打开 Session2, 开启事务
- session2.update(stu);
- // 提交事务,关闭 Session2
如果希望只有脱管对象改变了, Hibernate 才生成 update 语句,可以把映射文件中
(3)saveOrUpdate() 方法兼具 save() 和 update() 方法的功能,对于传入的对象, saveOrUpdate() 首先判断其是脱管对象还是临时对象,然后调用合适的方法。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持 PHPERZ!
来源: http://www.phperz.com/article/18/0103/357921.html