Hibernate6-Session缓存

在说Session缓存之前我们先修改一下我们的项目,在项目中添加两个方法

@Before
public void setUp() throws Exception {

}

@After
public void tearDown() throws Exception {

}

我的这两个方法是我使用ide生成的,我们使用单元测试测试某个方法的时候,junit会先帮我执 setup ()方法,当单元测试结束后junit会为我们执行tearDown()方法。当然方法名无所为,只要符合规定就可以随便写,主要的是@Before与@After这两个注解。我们将hibernateTest1()
方法中的代码剪切出来最终改造成这个样子

public class HibernateTest {
    private Session session = null;
    private Transaction transaction = null;

    @Before
    public void setUp() throws Exception {
        StandardServiceRegistry serviceRegistry = new StandardServiceRegistryBuilder().configure().build();
        SessionFactory sessionFactory = new MetadataSources(serviceRegistry).buildMetadata().buildSessionFactory();

        session = sessionFactory.openSession();
        transaction = session.beginTransaction();
    }

    @After
    public void tearDown() throws Exception {
        transaction.commit();
        session.close();
    }


    @Test
    public void hibernateTest1(){

    }
}

看明白了吧,在开始单元测试的时候帮我们获取一个Session对象并且开启一个事务,结束单元测试的时候,帮我们提交事务和关闭Session。

获取一条记录

    还记得我们在数据库中插入了一条 name字段值为张三的一条记录吧?我们现在来从数据库中取出来。哦,对了我忘了和大家说一下由于Mac电脑安装oracle安装比较麻烦,再者使用服务器还是有点不方便。以后我都会使用mysql社区版做演示。


Hibernate6-1.png

可以看出跟我们的预期效果是一样,发送了一条sql语句来查询数据,我们在使用get方法获取一遍数据


Hibernate6-2.png

    可以看出连续两条打印结果,但是只发送了一条sql语句来做查询。那么问题来了,为什么第一遍使用get方法获取数据需要发送sql语句,而第二次使用get方法获取数据则不需要发送sql语句了呢?
    这是因为,你第一次使用get方法的时候session会将你的要查询的数据拿出来返回给你之后,并且会在session缓存中保存一份,所以当你在同一个session中查询相同数据的时候,会先到自己的缓存中去找,如果有则返回给你,如果没有就去二级缓存中找,如果在没有数据彩绘发送sql语句从数据库中查找。

操作session缓存

    flush:当我们调用flush方法时,会将我们的增删改操作同步到数据库中。
    reflesh:强制同步缓存中的一个对象,就是说我们获取了数据,当我们过了一段时间要继续使用此数据的时候,但是不确定别人是否修改了此数据,所以使用此方法来重新获取一遍。当然我们也可以先使用evict 方法将对象从session缓存中移除,然后在使用    get 方法获取。
    clear:清空缓存。
    evict:从session缓存中移除某个对象。
    commit和flush区别:flush执行一系列sql语句,但不提交事务;commit方法先调用flush方法,然后提交事务。

理论

    一级缓存生命周期很短与Session生命周期一致,所以一级缓存也叫Session缓存或事物级缓存。为什么也叫事务级缓存呢,是因为增删改方法需要事务来完成,每次执行增删改后都要commit,所以也叫事务级缓存。
    位于缓存中的对象处于持久化状态,它和表中的相关记录对应,Session能够在某些时间点,按照缓存中持久话对象的属性变化来同步数据库中表的记录,这一过程称为清理缓存。也就是说当我们使用get方法查询出数据库某条记录的时候,这时对象就处于持久状态,例如:User user = session.get(User.class, 1);
我们从数据库中查询出主键为1的数据,这个时候user对象就处于持久状态。说白了就是通过session得到的对象都是持久状态对象。
    当我们修改了此对象的属性值,在使用session的方法进行提交和刷新缓存时,就会将你修改过的对象属性值同步到数据库表中,在提交或刷新缓存时的过程称为清理缓存。当然清理缓存不只是提交和刷新缓存,还有别的方法。

一级缓存实现原理:

    session缓存是由它的实现类SessionImpl中定义的一些集合属性构成的,原理是保证有一个引用在关联某个持久话对象,保持它的生命周期不会结束。 大家都知道java中有垃圾回收器也就是GC,如果我们长时间不使用某个对象,那么他会在一定时间内,被java的垃圾回收器从内存中清理掉,所以hibernate为了防止通过session获取的对象被清理掉,在session缓存中一直保持着此对象的引用,直到session被关闭。

一级缓存的作用

    减少数据库访问量,例如我们总是要获取数据,每次都要链接数据库查询,这样大大降低了运行效率,所以我们都会通过session缓存来获取数据。

配图
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容