Hibernate的三种状态

1.Hibernate对象的三种状态

Hibernate中有三种状态,对它的深入理解,才能更好的理解hibernate的运行机理,刚开始不太注意这些概念,后来发现它是重要的。对于理解hibernateJVMsql的关系有更好的理解。对于需要持久化的JAVA对象,在它的生命周期中有三种状态,而且互相转化。

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

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

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

×√

临时状态

Transient)

持久化状态

Persistent)

游离状态

Detached)

是否处于Session缓存中

×

×

数据库中是否有对应记录

×



三种状态的区别如下:

1. 当对象处于Transient时,只在内存中有一个对象,没ID,而且在缓存和数据库中没有;

2. 当对象处于save之后,内存、缓存都存在,有ID,而且当对象commit后数据库也存在;

3.当对象处于Detached时,内存、缓存、数据库都存在,并有ID,只是处于托管状态;


如图:

Hibernate的三种状态_第1张图片

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

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

游离对象有持久化对象关闭Session而转化而来,在内存中还有对象所以此时就变成游离状态了;

Hibernate和SQL的关系:

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

下面举例说明:
一,Session.save(user)运行机理。
1,把User对象加入缓存中,使它变成持久化对象;
2,选用映射文件指定的标识生成ID;
3,在Session清理缓存时候执行:在底层生成一个insert sql语句,把对象存入数据库;

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

二,Session.delete(user)运行过程。
如果user是持久化对象,则执行删除操作,同样底层数据库的执行条件是:在Session清理缓存时候;
如果user是游离对象:
1,将user对象和Session关联,使之成为持久化对象;
2,然后按照user 是持久化对象的过程执行;


       2. Session常用方法


 2. 1Session的两个方法区分:


得到Session的方法有如下两个:


openSession :每次都是新的Session,并且要手动close


getCurrentSession:从上下文找,如已有那么用已经有的Session,如没有,创建新的;不需要手动close;


ps.   上下文具体指的是hibernate.cfg.xml中的current_session_context_class:


  <property name=”current_session_context_class”>thread</property>


current_session_context_class有四个值,两个是常用的:


thread:在线程里找是否有已经存在的Session;(最常用)


jta:主要针对数据库分布式而用;(处理多个数据库事务)


2.2  delete :


delete(Object arg0); 当对象处于Detached后我们就可以使用delete进行删除其对象;


2.3  load :


ss2.load(Class arg0, Serializable arg1);  返回Object,强制转换下就OK了。


arg0:指的是你从数据库取出数据当成arg0类型处理;


arg1:指的是主键;


 2.4. get : 


ss2.load(Class arg0, Serializable arg1); 与Load方法一样可以实现取出数据的功能;


              注意:  load 和 get的区别!


2.4.1.  load返回的只是代理对象,只有当你真正使用对象内容的时候才会发出sql语句;而get则会直接从数据库


加载,立刻发出sql语句,不会延迟;


2.4.2. 当使用不存在的数据时,load不会报错,而get则肯定会报错;


       2.5 update:


update(Object arg0); 此种情况效率很低,因为默认将所有字段都进行更新;


so, 可以从数据库中获取到其对象后,直接set改变需要设置的属性,然后当session进行commit时,hibernate默


认同步数据库,如发现一致,不会发update的sql语句进行更新,当不一致的时候才会发sql的update的语句,当然


此时更新也是全部字段进行更新,效率仍然偏低;


如果需要单独更新一个字段,方法如下:


          2.5.1:修改注解(不灵活)当不需要更新时进行更新的属性,使用注解@Column(updatable=false)


updatable默认为true;


         2.5.2: 使用xml配置文件的话,可以使用dynamic-update


         2.5.3: HQL(EJBQL) (推荐)


  2.6 :saveOrUpdate:


saveOrUpdate(Object o);   自动选择save还是update;


  2.7: clear:    


clear(); 主要用于清除session的缓存;


  2.8: flush():


flush(); 强制让缓存内容与数据库内容做同步,默认当session的commit时才同步;


你可能感兴趣的:(Hibernate的三种状态)