Hibernate缓存机制之快照
(1)、快照是数据的副本
(2)、快照属于一级缓存
(3)、快照是在堆内存中的
(4)、快照的作用:保证数据一致性
当执行`session.getTransaction().commit()时,Hibernate同时会清理session的一级缓存(flush),也就是将堆内存中的数据与快照中的数据进行对比,如果不一致,则会执行同步(update)操作,若相同,则不执行update。
举个栗子
数据库中有一条数据:
测试代码:
public void testGet(){
//获取session 对象
Session session = HbnUtils.getSession();
//开启事务
session.beginTransaction();
try {
//执行get操作
Student student = session.get(Student.class, 1);
student.setName("王五");
System.out.println(student);
//提交事务
session.getTransaction().commit();
} catch (Exception e) {
//回滚事务
session.getTransaction().rollback();
e.printStackTrace();
}
}
日志信息
select
student0_.t_id as t_id1_0_0_,
student0_.t_name as t_name2_0_0_,
student0_.t_age as t_age3_0_0_,
student0_.t_score as t_score4_0_0_
from
t_student student0_
where
student0_.t_id=?
Student [id=1, name=王五, age=20, score=89.9]
Hibernate:
update
t_student
set
t_name=?,
t_age=?,
t_score=?
where
t_id=?
分析
我们都知道,执行了get方法之后,DB中的数据就加载到session缓存中来了,而执行student.setName("王五")本来应该只是改变了session缓存(堆内存)中的数据,为什么数据库的数据也改变了。
Student student = session.get(Student.class, 1);`
再来解剖一下
1)、将数据从DB中取出来
2)、将数据转变成对象,并存入堆内存中
3)、将对象的id放入session缓存map的key中,将对象的引用放入session缓存map的 value中,这就纳入session管理了
4)、将对象的详情放入到“快照”中
当执行了`session.getTransaction().commit();时,Hibernate为了保证数据的一致性,Hibernate会清理session的一级缓存(flush),也就是将堆内存中的数据(已经纳入session管理的数据)与快照中的数据进行对比,如果不一致,则会执行同步(update)操作,若相同,则不执行update。
由于在commit()前,我们执行了student.setName("王五");
导致堆内存中是数据与快照中的数据不一致,所以它执行了update,以便保证数据的一致性。
为什么需要快照
通过上面的分析知道了快照的根本作用是保证数据一致性,保证数据一致的另一种做法是,commit之前把堆内存中的数据直接与数据库中的对应记录进行对比,显而易见这样的效率是灰常低下的,而采用快照技术,因为快照是一定和数据库中记录一致的,快照也在堆内存中,所以速度不是一般的快。