hibernate的save和update初步理解

hibernate状态知识点补充:Hibernate中的对象有三种状态: 瞬时状态 (Transient),持久状态 (Persistent),

1. 脱管状态 (Detached)

1. 瞬时状态 (Transient)

由 new 命令开辟内存空间的 Java 对象,也就是平时所熟悉的普通 Java 对象。

如: Student stu = new Student();

瞬时对象特点:

(1) 不和 Session 实例关联

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

2. 持久状态 (Persistent)

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

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

瞬时对象转为持久对象:

(1) 通过 Session 的 save() 和 saveOrUpdate() 方法把一个瞬时对象与数据库相关联,这个瞬时对象就成为持久化对象。

(2) 使用 fine(),get(),load() 和 iterater() 待方法查询到的数据对象,将成为持久化对象。

持久化对象的特点:

(1) 和 Session 实例关联

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

3.  脱管状态 (Detached)

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

脱管对象特点:

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

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

持久对象转为脱管对象:

当执行 close() 或 clear(),evict() 之后,持久对象会变为脱管对象。

瞬时对象转为持久对象:

通过 Session 的 update(),saveOrUpdate() 和 lock() 等方法,把脱管对象变为持久对象。

       接下来,对save、update和saveOrUpdate方法进行简单的描述,不是很清楚hibernate作者为什么分别写了以上三个方法,通过saveOrUpdate方法就可以实现保存和更新操作,其实save和update和saveOrUpdate到了底层,都是调用SessionImpl类,  public Serializable save(String entityName, Object object) throws HibernateException {

returnfireSave(newSaveOrUpdateEvent(entityName,object,this));

  }

     以下信息我以save方法为例,根据hibernate源码,具体的大致原理是:

     session的 Save() 方法将瞬时对象保存到数据库,对象的临时状态将变为持久化状态。当对象在持久化状态时,

它一直位于Session的缓存中,对它的任何操作在事务提交时都将同步到数据库,因此,

对一个已经持久的对象调用save()或update()方法是没有意义的

    session.save(eo);

     ---->将对象标示为persistent状态,放到Hiberante Session级别(当前线程)队列中,产生一个 insertObjectAction,

 代码如下:AbstractSaveEventListener类performSaveOrReplicate方法

source.getActionQueue().addAction( new EntityInsertAction( id, values, entity, version, persister, source )中

  真正执行sql是session事务提交时会执行frush操作,对应到具体操作:EntityInsertAction类public void execute  ()  throws HibernateException()方法执行sql,

再save的时候,比对eo是否为persistent状态,若是,不产生insertObjectAction;

比对缓存中数据的状态,若变化产生一个updateObjectAction

  hibernate 什么时候提交数据(flush缓存中的)数据到DB ? 更新session缓存,执行sql语句。

   补充:修改数据的时候hibernate对比save操作会查找查找出EO实体表的数据,把属性值放在一个数组中,然后和修改后的EO各个属性值进行对比,会有一个status来标识eo的状态,hibernate会维护实体的数据的状态信息,用来确定是新增数据还是更新数据。