hibernate中持久化对象的生命周期(三态:自由态,持久态,游离态 之间的转换)

  在hibernate中有三种状态:瞬时态(Transient)、 持久态(Persistent)、脱管态(Detached)。处于持久态的对象也称为PO(Persistence Object),瞬时对象和脱管对象也称为VO(Value Object)。这三种状态也可以称作:临时状态、持久状态、游离状态。
一、三种状态介绍:
(1) 瞬时状态:
  由new操作符创建,且尚未与Hibernate Session关联的对象被认定为瞬时(Transient)的。

  瞬时(Transient)对象不会被持久化到数据库中,也不会被赋予持久化标识(identifier)。如果瞬时(Transient)对象在程序中没有被引用,它会被垃圾回收器(garbage collector)销毁。使用Hibernate Session可以将其变为持久(Persistent)状态。(Hibernate会自动执行必要的SQL语句)。
  瞬时状态的特点有:
  1、 与数据库中的记录没有任何关联,也就是没有与其相关联的数据库记录。
  2、 与Session没有任何关系,也就是没有通过Session对象的实例对其进行任何持久化的操作。

User user=new User();      //user是一个瞬时对象,在数据库的表中是没有记录和该对象相对应的。和session没有关系。
user.setName(“ddd”);
user.setBirthday(new Date());

session.save(user); //持久化状态

(2) 持久状态:
  持久(Persistent)是实例在数据库中有对应的记录,并拥有一个持久化标识(identifier)。

  持久(Persistent)的实例可能是刚被保存的,或刚被加载的,无论哪一种,按定义,它存在于相关联的Session作用范围内。

  Hibernate会检测到处于持久(Persistent)状态的对象的任何改动,在当前操作单元(unit of work)执行完毕时将对象数据(state)与数据库同步(synchronize)。开发者不需要手动执行Update。将对象从持久(Persistent)状态变成瞬时(Transient)状态同样也不需要手动执行delete语句。
  持久对象具有如下特点:
  1、 和session实例关联;
  2、 在数据库中有与之关联的记录。
  3、 Hibernate会根据持久态对象的属性的变化而改变数据库中的相应记录。
  举例:

        Session session = factory.openSession();    
        Transaction tx = session.beginTransaction();    
   
        session.save(stu); // persistent持久化状态    
        System.out.println(stu);    
   
        tx.commit();    
        session.close();    //执行close()方法之后,就会由持久对象转换成脱管对象
   
        System.out.println(stu); // 脱管对象

(3)脱管状态:
  与持久(Persistent)状态对象关联的Session被关闭后,对象就变为脱管(Detached)的。对脱管(Detached)对象的引用依然有效,对象可继续被修改。

  脱管(Detached)对象如果重新关联到某个新的Session上,会再次转变为持久(Transistent)的(在脱管Detached其间的改动将被持久化到数据库)。这个功能使得一种编程模型,即中间会给用户思考时间(user think-time)的长时间运行的操作单元(unit of work)。
  脱管对象拥有数据库的识别值,可通过update()、saveOrUpdate()等方法,转变成持久对象。
  脱管对象具有如下特点:
  1、 本质上与瞬时对象相同,在没有任何变量引用它时,JVM会在适当的时候将它回收;
  2、 比瞬时对象多了一个数据库记录标识值。
  3、 不在于Session相关联。
  4、 脱管对象一定是由持久态对象转化而来。

总结:

  1、当一个对象被new了以后此对象处于瞬时态(Transient);
  2、然后对此对象执行session的save() 或者saveOrUpdate()方法后,此对象被放入session的一级缓存进入持久态.
  3、当再对此对象执行evict()/close()/clear()的操作后此对象进入游离态(Detached)
  4、游离态(Detached)和瞬时态(Transient)的对象由于没有被session管理会在适当的时机被java的垃圾回收站(garbage)回收.
  5、执行session的get()/load()/find()/iternte()等方法从数据库里查询的到的对象,处于持久态(Persistent).  
  6、当对数据库中的纪录进行update()/saveOrUpdate()/lock()操作后游离态的对象就过渡到持久态 
  7、处于持久态(Persistent)与游离态(Detached)的对象在数据库中都有对应的记录.
  8、瞬时态(Transient)与游离态(Detached)的对象都可以被回收可是瞬时态的对象在数据库中没有对应的纪录,而游离态的对象在数据库中有对用的纪录。

二、游离对象和临时对象异同:

  两者都不会被Session关联,对象属性和数据库可能不一致;

  游离对象由持久化对象关闭Session而转化而来,在内存中还有对象,所以此时就变成游离状态了;
三、Hibernate和SQL的关系:

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

  下面举例说明:
  1、Session.save(user)运行机理。
  (1)把User对象加入缓存中,使它变成持久化对象;
  (2)选用映射文件指定的标识生成ID;
  (3)在Session清理缓存时候执行:在底层生成一个insert sql语句,把对象存入数据库;
  注意:在你执行Session.save(user)后,在Session清理缓存前,如果你修改user对象属性值,那么最终存入数据库的值将是最后修改的值;此过程中ID不能被修改;
  2、Session.delete(user)运行过程。
  (1)如果user是持久化对象,则执行删除操作,同样底层数据库的执行条件是:在Session清理缓存时候;
  (2)如果user是游离对象:

  第一步:将user对象和Session关联,使之成为持久化对象;

  第二步:然后按照user是持久化对象的过程执行;

四、三态之间的转换方法:

  如何成为自由态?

  对象通过构造方法成为自由态;

  持久态和游离态则通过session的delete方法成为自由态

  如何成为持久态?

  对象可以由session的load或get方法直接成为持久态;

  自由态对象可以通过save,saveOrUpdate或persist方法成为持久态;

  游离态对象则可以通过update,saveOrUpdate成为持久态

  如何成为游离态?

  游离态只能由持久态转换而来,通过close或clear方法实现。

五、几种转换方法的对比:

  1、get 与load

  都是从数据库中加载数据封装为java对象,使得java对象从自由态直接变为持久态;

  但是有两点区别:

  (1)get返回对象可以为null,load返回值则始终不为null,找不到时会抛异常

  (2)get即时执行insert,而load则是在使用此对象时才执行insert

  2、save,update与saveOrUpdate

  save是将自由态转为持久态,而update是将游离态转为持久态,saveOrUpdate可以说是两者的综合,它执行时先判断对象的状态(主要是通过有无主键判断的),若是自由态,则save,若是游离态,则update

  3、save与persist

  两者都是将对象由自由态转为持久态,但返回值不同:save返回主键值,而persist不返回

  4、saveOrUpdate与merge

  两者都是将自由态或游离态对象与数据库关联,但merge不改变对象的原有状态

  此外,对clear与flush方法也作介绍。

  clear是将session中的对象全部变为游离态,是对象由持久态变为游离态的一种方法(另外一种是关闭session);

  flush方法时为了使update操作能即时进行(正常情况下,只有在事务关闭时才进行update操作)。

原文地址:https://www.cnblogs.com/goloving/p/8268709.html