关于Hibernate的commit、flush方法与对象的三种状态

来源:互联网 发布:网络教育英文翻译 编辑:程序博客网 时间:2024/05/20 03:40
在Hibernate中有三种状态,只有对它的深入理解,才能更好的理解hibernate的运行机理,刚开始不太注意这些概念,后来发现它是重要的。

对于理解hibernate,JVM和sql的关系有更好的理解。

对于需要持久化的JAVA对象,在它的生命周期中有三种状态,而且互相转化。

Hibernate三种状态之一:临时状态(Transient):用new创建的对象,它没有持久化,没有处于Session中,处于此状态的对象叫临时对象;

Hibernate三种状态之二:持久化状态(Persistent):已经持久化,加入到了Session缓存中。如通过hibernate语句保存的对象。处于此状态的对象叫持久对象;

Hibernate三种状态之三:游离状态(Detached):持久化对象脱离了Session的对象。如Session缓存被清空的对象。特点:已经持久化,但不在Session缓存中。处于此状态的对象叫游离对象;

Hibernate三种状态中游离对象和临时对象异同:

两者都不会被Session关联,对象属性和数据库可能不一致;游离对象由持久化对象关闭Session而转化而来,在JVM中还有对象所以此时就变成游离状态了;

Hibernate和SQL的关系:在操作了hibernate的方法如save()等后,并没有直接生成sql语句,去操作数据库,而是把这些更新存入Session中,只有Session缓存要被commit时,底层的sql语句才能执行,数据存入数据库;

注意:在你执行Session.save(user)后,在Session清理缓存前,如果你修改user对象属性值,那么最终存入数据库的值将是最后修改的值;此过程中ID不能被修改;

下面我们来看代码:

session.beginTransaction();session.save(t);session.getTransaction().commit();

代码很简单,向数据库表存入一条数据,我们将Hibernate的”show_sql”属性设置为true,控制台信息如下:

Hibernate: select max(id) from Teacher
Hibernate: insert into Teacher (age, name, id) values (?, ?, ?)

这里也许看不出什么端倪来,现在在session.save(t)后面加一行代码,输出这个teacher的OID,System.out.println(teacher.getId()),再次运行程序,控制台输出为:

Hibernate: select max(id) from Teacher
5
Hibernate: insert into Teacher (age, name, id) values (?, ?, ?)

这里可以说明insert语句是在commit时才执行到数据库中

我们把session.getTransaction().commit()语句换成session.flush(),然后我们看控制台信息:

Hibernate: select max(id) from Teacher
6
Hibernate: insert into Teacher (age, name, id) values (?, ?, ?)

看样子跟没改之前一样,但是我们查一下数据库,没有id为6的这条数据,这说明数据库执行了这条insert语句,因为没有commit数据,又callback回来了。只是通过flush把它打印出来,而调用commit方法,该方法自动调用flush方法。

flush方法的主要作用就是清理缓存,强制数据库与Hibernate缓存同步,以保证数据的一致性。它的主要动作就是向数据库发送一系列的sql语句,并执行这些sql语句,但是不会向数据库提交。而commit方法则会首先调用flush方法,然后提交事务。这就是为什么我们仅仅调用flush的时候记录并未插入到数据库中的原因,因为只有提交了事务,对数据库所做的更新才会被保存下来。因为commit方法隐式的调用了flush,所以一般我们都不会显示的调用flush方法。

一个基本的插入流程就出来了:

  1. 判断所要保存的实例是否已处于持久化状态,如果不是,则将其置入缓存,并根据定义的generator生成一个id号
  2. 根据所要保存的实例计划一条insert sql语句,注意只是计划,并不执行
  3. 事务提交时执行之前所计划的insert语句