博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Hibernate的session一级缓存
阅读量:5094 次
发布时间:2019-06-13

本文共 2067 字,大约阅读时间需要 6 分钟。

一级缓存是Session周期的,当session创建的时候就有,当session结束的时候,缓存被清空

当缓存存在的时候,每次查询的数据,都会放在缓存中,如果再次查询相同的数据,则不会再次查询数据库,可以有效的减少数据库的访问量。

但是,session的生命周期很短,当session创建,进行数据库操作后,就会被关闭,同样的,缓存就会被清空。如果是在javaweb中,session的生命周期,就是浏览器向服务器的一次请求。所以,后面hibernate会有相应的二级缓存

 

 

操作缓存的几个方法介绍

一:flush()

flush方法会同步缓存与数据库中的数据,就是说,如果从数据库中得到一条数据,也就是一个对象,如果在缓存中,对于这个对象,进行了数据的更改,那么在使用flush()方法的时候,将会把在缓存中修改的数据,同步到数据库中,有可能是删除,也有可能是修改,或者插入。

注:在事务的提交的时候,会自动执行flush()方法,来同步缓存与数据库中的数据。

注意:就算没有执行Transaction 的 commit() 方法,或者手动调用fulsh()方法,也有可能会被调用flush()方法原因如下

 

1). 执行 HQL 或 QBC 查询, 会先进行 flush() 操作, 以得到数据表的最新的记录

* 2). 若记录的 ID 是由底层数据库使用自增的方式生成的, 则在调用 save() 方法时, 就会立即发送 INSERT 语句.
* 因为 save 方法后, 必须保证对象的 ID 是存在的!

 

二:refresh()

refresh方法会同步缓存与数据库中的数据,就是把当前数据库中最新的数据给缓存中的对象下面给出两个例子!

例子1:在高并发的情况,当我已经拿到一行表数据的时候,并且已经封装成了一个对象。这个时候,数据库中的这一行记录被其它的线程改变了数据,那么就是说,我现在的这个对象,已经不是数据库中的最新的数据了,如果使用session.refresh()方法,则会把数据库中最新的数据同步到缓存中!

 

例子2:这个要给出代码来示例一下,虽然说代码没什么特定的作用,但是还是可以看出一些问题的

 

//从数据库中拿到数据,并封装成对象        News ne = (News) session.get(News.class, 1);        //这里对这个持久化类进行了修改其成员变量,如果不出意外的话,下面事务提交的时候,就会把这条数据自动提交到数据库中        ne.setAuthor("刘军");        //同样的,我打印这个对象,其实这个时候,输出的已经不是数据库中的数据了,因为我对这个对象作了改变        System.out.println(ne);        //但是如果我对这个对象作refresh()操作的话,就会把数据库中最新的数据同步到缓存对象中        session.refresh(ne);        //打印出来的数据,又是一开始的数据了,也就是刚拿到这个对象的时候的数据        System.out.println(ne);

经过上面的两个例子,就可以发现,refresh()方法,是把数据库中的数据,同步到缓存中!

注:由于数据库的事务机制,在同一个事务中,会保证操作的原子性,所以在此外改的数据,refresh()方法,是刷新不到的,并不是这个方法有问题,而是数据库的隔离机制,Oracle有两种隔离机制,mysql有四种隔离机制,只要把mysql设置成可重复读,就可以查看到最新的数据了!

 

 

三:clear()

这个方法,将会把session中当前的全部缓存,都给清空!

 

 

四:persist()

它和save()方法一样,都会执行insert操作,但是为什么hibernate会有两个保存数据的方法呢?自然是有区别的,下面给出两个方法的区别!

1:save方法,当执行insert的时候,临时状态的对象,应该是没有id的,如果这个时候,对象被插入了id,那么其实hibernate插入数据的时候,仍然会以主键生成策略为准,所以人为设置的id,是没有作用的,当执行完save方法后,id会重新变成hibernate生成的,或者sql数据库生成的id。

2:persist方法,同样的,它也是一个保存至数据库的一个方法,但是使用persist方法的时候,hibernate会检查这个对象是否已经有主键id了,如果已经有,则不会执行insert方法,相反的会抛出异常!

 

两个方法的作用比较,使用save方法,可以把一个对象多次保存,使用persist对象,可以保证这个数据,只能插入一次,因为只要执行了一次,这个对象就会有id值,只要有了id值,就不能再执行persist对象,所以可以依据自己的情况来使用这两个方法!

 

转载于:https://www.cnblogs.com/zhuxiaojie/p/4943892.html

你可能感兴趣的文章
Blender Python UV 学习
查看>>
window添加右键菜单
查看>>
入手腾龙SP AF90mm MACRO
查看>>
ORACLE 递归查询
查看>>
20172315 2017-2018-2 《程序设计与数据结构》实验三报告
查看>>
别把SEO当苦力活,做优化要讲究策略
查看>>
Django项目:CRM(客户关系管理系统)--41--33PerfectCRM实现King_admin编辑整张表限制
查看>>
关于时间
查看>>
面向对象 阶段性总结
查看>>
[Android] 开发第十天
查看>>
[html]window.open 使用示例
查看>>
.NET下使用socket.io随笔记录
查看>>
操作~拷贝clone()
查看>>
通过this()调用有参构造方法
查看>>
Java开发中的23种设计模式
查看>>
jQuery源码分析(2) - 为什么不用new jQuery而是用$()
查看>>
jQuery源码分析(3) - 判断传入对象是否为function或array
查看>>
[转]【EL表达式】11个内置对象(用的少) & EL执行表达式
查看>>
ArrayList对象声明& arrayList.size()
查看>>
并发编程 线程
查看>>