一、Hibernate缓存简介:
Session接口是Hibernate向应用程序提供的操纵数据接口的最主要接口,它提供了基本的保存、更新、删除和加载Java对象的方法。
Session具有一个缓存,位于缓存中的对象称为持久化对象,Session能够在某些时间点来同步更新数据库,这一过程即为清理缓存。
Hibernate把对象分为4种状态:持久化状态、临时状态、游离状态和删除状态。
二、Session的缓存:
Session的缓存由一系列Java集合构成的。
1、Session缓存的作用:
A、减少访问数据库的频率。
B、当缓存中的持久化对象之间存在循环关联关系时,Session保证不会出现访问对象图的死循环,以及由死循环引起的JVM堆栈溢出异常。
C、保证数据库中的相关记录和缓存中的相应对象保存同步。Session通过在清理缓存时进行脏检查来同步。
2、脏检查及清理缓存的机制:
当一个对象被加入到Session缓存中时,Session会为该对象的值类型的属性复制一份快照。在清理缓存进行脏检查时,就是拿对象的当前属性和它的快照进行对比,据此判断是否发生了变化。
另外,当对象属性发生变化时,Session并不会立即清理缓存和执行相关的SQL操作,而是在特定的时间才清理缓存。这样的好处就是Session能够把几条相关的SQL语句合并为一条SQL,从而减少访问数据库的次数。
Session清理缓存时,按照以下顺序执行SQL语句(了解):
?
1 2 3 4 5 6 |
|
Session会在以下时间点清理缓存:
?
1 2 3 |
|
例外情况:如果对象使用native生成器来生成OID,name当调用Session的 save()方法保存对象时,会立即执行insert语句。
也可以通过Session.setFlushMode()方法来显式的设定清理缓存的时间点。
多数情况下应用程序不需要显式的调用flush()方法,以下常见除外:
A、插入、删除或更新某个持久化对象会引发数据库中触发器的时候。
B、在应用程序中混合使用Hibernate
API和JDBC API的时候。
C、JDBC驱动程序不健壮,导致Hibernate在自动清理缓存模式下无法正常工作的时候。
三、Java对象在Hibernate持久化层的状态:
1、状态描述:
A、临时状态(transient):刚用new语句创建,还没有被持久化,并且不处于Session的缓存中。
B、持久化对象(persistent):已经被持久化,并且加入到Session缓存中。
C、删除状态(delete):不再处于Session的缓存中,并且Session已经计划将其从数据库中删除。
D、游离状态(detached):已经被持久化,但是不再处于Session的缓存中。
2、状态转换图:
3、临时对象的特征:
A、在使用代理主键的情况下,OID通常为NULL。
B、不处于Session的缓存中,不被任何一个Session实例关联。
C、在数据库中没有对应的记录。
四、持久化对象的特征:
1、OID不为null。
2、位于一个Session实例的缓存中。持久化对象总是被一个Session实例关联。
3、持久化对象和数据库中的相关记录对应。
4、Session在清理缓存时,会根据持久化对象的属性变化,来同步更新数据库。
五、被删除对象的特征:
1、OID不为null。
2、从一个Session实例的缓存中删除。
3、被删除对象和数据库中的相关记录对应。
4、Session已经计划将其从数据库中删除。
5、Session在清理缓存时,会执行SQL
delete语句,删除数据库中的相应记录。
6、一般情况下,应用程序不应该再使用被删除的对象。
六、游离对象的特征:
1、OID不为null。
2、不再位于Session的缓存中。
3、游离对象是由持久化对象转变过来的,因此在数据库中可能还存在与它相对应的记录。
游离对象和临时对象的相同之处:两者都不被Session关联,因此不会保证它们的属性变化与数据库保存同步。
游离对象和临时对象的不同之处:游离对象是由持久化对象转变过来的,因此数据库中可能存在对应的记录,而临时对象在数据库中没有与之对应的记录。
游离对象与被删除对象的相同之处:两者都不位于Session的缓存中,并且数据库中都可能存在对应的记录。
游离对象与被删除对象的不同之处:游离对象与Session完全脱离关系,而对于被删除对象,Session会计划将其从数据库中删除,在清理缓存时会执行删除操作。
七、Session接口的详细用法:
1、Session的save()和persist()方法:
save()方法使一个临时对象转变为持久化对象。注意应该禁止应用程序修改OID。
persist()方法和save()方法类似,区别在于该方法是在Hibernate3版本中财出现的。且如果在事务边界以外调用persist()方法时,不会执行SQL
insert语句。
2、Session的load()和get()方法:
这两个方法都能根据给定的OID从数据库中加载一个持久化对象。
区别在于:
A、当数据库中不存在与OID对应的记录时,load()方法抛出OjbectNotFoundException,而get()方法返回null。
B、两者采用不同的检索策略(检索策略后续描述)。load()方法根据<class>元素的lazy属性决定是否进行懒加载,而get()方法会忽略lazy属性,采用立即检索策略。
使用场景:
A、如果加载一个对象的目的是访问它的各个属性,可以用get()方法。
B、如果加载一个对象时为了删除它,或者威力建立和别的对象的关联,可以用load()方法。
3、Session的update()方法:
update()方法使一个游离对象转变为持久化对象,并计划执行一条update语句。
如果希望Session仅在修改了对象的属性时才执行update语句,可以把<class>元素的select-before-update设置为true。
当update()方法关联一个游离对象时,如果在Session缓存中已经存在相同OID的持久化对象,会抛出异常。
4、Session的saveOrUpdate()方法:
saveOrUpdate()方法同时包含了save()和update()方法。如果传入的参数是临时对象,调用save()方法;如果传入的参数是游离对象,就调用update()方法;如果传入的是持久化对象,就直接返回。
满足下列任何一个条件,Hibernate就会认为该对象时临时对象:
A、Java对象的OID为null。
B、Java对象具有version版本控制属性并且取值为null(不明白)。
C、在映射文件中为<id>元素设置了unsaved-value属性,并且Java对象的OID与这个属性值匹配。
D、在配置文件中为version版本控制属性设置了unsaved-value属性,并且Java对象的version版本控制属性的取值与该属性值匹配。
E、为Hibernate的Interceptor提供了自定义的实现,并且Interceptor实现类的isUnsaved()方法返回Boolean.TRUE。
5、Session的merger()方法:
merge()方法能够把一个游离对象的属性复制到一个持久化对象中。
6、Session的delete()方法:
delete()方法用于从数据库中删除一个Java对象。既可以删除持久化对象,也可以删除游离对象。
7、Session的replicate()对象:
该方法能够把一个数据库中的对象复制到另一个数据库中。(暂不关注)
八、级联操作对象图:
1、cascade属性表:
(声明:该文所有内容均来自《精通Hibernate:Java对象持久化技术详解》[孙卫琴
电子工业出版社] 一书。该文的目的仅仅作为学习笔记。若需要转载,请注明原书相关信息。)