七、哪些方法支持缓存
*get()
*load()
*iterate() ( 查询实体对象 )
save()
查询缓存只对 query.list() 起作用
一级缓存测试:
1 . Load 测试 : 在同一个 session 中发出两次 load 查询( 1 )
2 . Get 测试 : 在同一个 session 中发出两次 get 查询( 1 )
3 . iterate 测试 : 在同一个 session 中发出两次 iterator 查询( 1 ( id ) +N , 1 ( id ))
4 . Iterate 查询属性测试 : 同一个 session 中发出两次查询属性( 2 次, iterate 查询普通属性,一级缓存不会缓存,所以会发出 sql )
5 .同一个 session 中先 save ,再发出 load 查询 save 过的数据 --save 是使用缓存的
6 .同一个 session 中先调用 load 查询,然后执行 sessio.clear() 或 session.evict() ,再调用 load 查询( 2 次)
sessio.clear() 或 session.evict() 可以管理一级缓存,一级缓存无法取消,但可以管理 . 上面的语句都会发出 sql 因为一级缓存中的实体被清除了
7 .向数据库中批量加入 1000 条数据
// 每一定条数据就强制 session 将数据持久化,同时清除缓存,避免大量数据造成内存溢出
开启二级缓存测试:
1 .开启两个 session 中发出两次 load 查询( get 与 load 一样, 1 次) ,
2 .开启两个 session ,分别调用 load ,再使用 sessionFactory 清楚二级缓存( 2 次)
3 .一级缓存和二级缓存的交互
session.setCacheMode(CacheMode.GET); // 设置成 只是从二级缓存里读 , 不向二级缓存里写数据 ( 2 )
session.setCacheMode(CacheMode.PUT); // 设置成只是向二级缓存里写数据,不读数据 ( 2 次)
开启 hibernate 查询缓存测试:
1 . 开启查询缓存,关闭二级缓存,开启一个 session ,分别调用 query.list (查询属性)( 1 次)
2 . 开启查询缓存,关闭二级缓存,开启两个 session ,分别调用 query.list (查询属性)
( 1 次)第二次没有去查询数据库,因为查询缓存生命周期与 session 生命周期无关
3 . 开启查询缓存,关闭二级缓存,开启两个 session ,分别调用 query.iterate (查询属性)
( 2 次)第二去查询数据库,因为查询缓存只对 query.list() 起作用,对 query.iterate() 不起作用 , 也就是说 query.iterate() 不使用查询缓存
4 . 关闭查询缓存,关闭二级缓存,开启两个 session ,分别调用 query.list (查询实体对象) 第二去查询数据库,因为 list 默认每次都会发出查询 sql
5 . 开启查询缓存,关闭二级缓存,开启两个 session ,分别调用 query.list (查询实体对象) 第二去查询数据库时,会发出 N 条 sql 语句,因为开启了查询缓存,关闭了二级缓存,那么查询缓存会缓存实体对象的 id ,所以 hibernate 会根据实体对象的 id 去查询相应的实体,如果缓存中不存在相应的实体,那么将发出根据实体 id 查询的 sql 语句,否则不会发出 sql ,使用缓存中的数据
6 . 开启查询缓存,开启二级缓存,开启两个 session ,分别调用 query.list (查询实体对象) 第二不会发出 sql ,因为开启了二级缓存和查询缓存,查询缓存缓存了实体对象的 id 列表, hibernate 会根据实体对象的 id 列表到二级缓存中取得相应的数据