1.Hibernate对象的三种状态
sql
在Hibernate中有三种状态,对它的深刻理解,才能更好的理解hibernate的运行机理,刚开始不太注意这些概念,后来发现它是重要的。对于理解hibernate,JVM和sql的关系有更好的理解。对于须要持久化的JAVA对象,在它的生命周期中有三种状态,并且互相转化。
数据库
1, 临时状态(Transient):用new建立的对象,它没有持久化,没有处于Session中,处于此状态的对象叫临时对象; 缓存
2, 持久化状态(Persistent):已经持久化,加入到了Session缓存中。如经过hibernate语句保存的对象。处于此状态的对象叫持久对象; session
3, 游离状态(Detached):持久化对象脱离了Session的对象。如Session缓存被清空的对象。
特色:已经持久化,但不在Session缓存中。处于此状态的对象叫游离对象; 分布式
×√ spa |
临时状态 hibernate (Transient) 线程 |
持久化状态 代理 (Persistent) xml |
游离状态 (Detached) |
是否处于Session缓存中 |
× |
√ |
× |
数据库中是否有对应记录 |
× |
√ |
√ |
三种状态的区别以下:
1. 当对象处于Transient时,只在内存中有一个对象,没ID,并且在缓存和数据库中没有;
2. 当对象处于save以后,内存、缓存都存在,有ID,并且当对象commit后数据库也存在;
3.当对象处于Detached时,内存、缓存、数据库都存在,并有ID,只是处于托管状态;
如图:
游离对象和临时对象异同:
二者都不会被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时才同步;