查詢快取的使用,主要是為了提高查詢存取速度。這篇文章主要介紹了MyBatis查詢快取,需要的朋友可以參考下
查詢快取的使用,主要是為了提高查詢存取速度。將使用者對相同數據的重複查詢過程簡化,不再每次均從資料庫查詢取得結果數據,從而提高存取速度。
MyBatis的查詢快取機制,根據快取區的作用域(生命週期)可分割為兩種:一級快取與二級快取
一、一層級查詢緩存
MyBatis一級快取是基於org.apache.ibatis.cache.impl.PerpetualCache類別的HashMap本地緩存,其作用域是Sqlsession。在同一個Sqlsession中兩次執行相同的sql語句,第一次執行完畢後,會將查詢結果寫入到快取中,第二次會從快取中直接獲取數據,而不再到資料庫中進行查詢,從而提高查詢效率。
當一個Sqlsession結束後,該Sqlsession中的一級快取也就不存在了。 MyBatis預設一級快取是開啟狀態,且不能關閉。
1.一級快取的存在性證明
#測試類別:
//证明一级缓存的存在 @Test public void test01(){ //第一次查询 Student student = dao.selectStudentById(2); System.out.println(student); //第二次查询 Student student2 = dao.selectStudentById(2); System.out.println(student2); }
mapper:
<mapper namespace="com.hcx.dao.IStudentDao"> <select id=selectStudentById resultType="com.hcx.beans.Student"> select * from student where id=#{id} </select> </mapper>
控制台:
執行完後,發現只執行了一次從DB中的查詢,第二次的結果是直接輸出的。說明,第二次是從Sqlsession快取讀取的。
2.從快取讀取資料的依據是sql的id
##一級快取快取的是相同sql映射id的查詢結果,而非相同sql語句的查詢結果。因為MyBatis內部對於查詢緩存,無論是一級查詢還是二級查詢,其底層均使用一個hashmap實作:key為sql的id相關內容,value為從資料庫中查詢出的結果。 mapper:<mapper namespace="com.hcx.dao.IStudentDao"> <select id=selectStudentById resultType="com.hcx.beans.Student"> select * from student where id=#{id} </select> <select id="selectStudnetById2" resultType="com.hcx.beans.Student"> select id,name,age,score,birthday from student where id=#{id} </select> </mapper>dao介面:
public interface IStudentDao { Student selectStudentById(int id); Student selectStudentById2(int id); }測試類別:
//证明从一级缓存中读取数据的依据: //MyBatis:sql的id+sql语句 //hibernate:查询结果对象的id @Test public void test02(){ Student student = dao.selectStudentById(2); System.out.println(student); Student student2 = dao.selectStudentById2(2); System.out.println(student2); }控制台:查看控制台,發現第二次查詢結果與第一次的完全相同,但第二次查詢並沒有從快取中讀取數據,而是直接從DB中進行的查詢。這是因為從快取讀取資料的依據是查詢sql的映射id,而非查詢結果。
3.增刪改對一級查詢快取的影響
增刪改操作,無論是否進行提交Sqlsession.commit( ),均會清空一級查詢緩存,使查詢再次從DB中select。 測試類別:@Test public void test03(){ Student student = dao.selectStudentById(2); System.out.println(student); //增删改操作都会清空一级缓存,无论是否提交 dao.insertStudent(new Student("赵六",26,96.6)); Student student2 = dao.selectStudentById(2); System.out.println(student2); }控制台:
二、內建二級查詢快取
MyBatis查詢快取的作用域是根據映射檔案mapper的namespace劃分的,相同namespace的mapper查詢資料存放在同一個快取區域。不同namespace下的資料互不干擾。 無論是一級快取還是二級緩存,都是依照namespace進行分別存放的。但一、二級快取的不同之處在於,Sqlsession一旦關閉,則Sqlsession中的資料將不存在,即一級快取就不復存在。而二級快取的生命週期會與整個應用程式同步,與Sqlsession是否關閉無關。 使用二級快取的目的,不是共享數據,因為MyBatis從快取讀取資料的依據是sql的id,而非查詢出的物件。所以,二級快取中的資料不是為了在多個查詢之間共享(所有查詢中只要查詢結果中存在該物件的,就直接從快取中讀取,這是對資料的共享,hibernate中的快取就是為了共享,但MyBatis不是),而是為了延長該查詢結果的保存時間,提高系統效能。1.二級快取用法
二級快取的使用只需要完成兩個步驟:序列化實體#在mapper映射檔中加入e20c50086bb6799ec27a6a73752fc01b標籤1.實體序列化
要求查詢結果所涉及的實體類別要實作java.io .Serializable介面。若該實體類別存在父類,或其具有域屬性,則父類與域屬性類別也要實作序列化介面。public class Student implements Serializable{ private Integer id; private String name; private int age; private double score; }
2.mapper映射檔中新增e20c50086bb6799ec27a6a73752fc01b標籤
在mapper映射檔中的7299502c94b6d6b501b3b016f577e505標籤中新增e20c50086bb6799ec27a6a73752fc01b子標籤
#
<mapper namespace="com.hcx.dao.IStudentDao"> <cache/> <select id=selectStudentById resultType="com.hcx.beans.Student"> select * from student where id=#{id} </select> </mapper>
3.二級快取的設定
为e20c50086bb6799ec27a6a73752fc01b标签添加一些相关属性设置,可以对二级缓存的运行性能进行控制。若不指定设置,则均保持默认值。
<cache eviction="IFIO" flushInterval="10800000" readOnly="true" size="512"/>
eviction:逐出策略。当二级缓存中的对象达到最大值时,就需要通过逐出策略将缓存中的对象移出缓存。默认为LRU。常用的策略有FIFO和LRU
flushInterval:刷新缓存的时间间隔,单位毫秒。这里的刷新缓存即清空缓存。一般不指定,即当执行增删改时刷新缓存。
readOnly:设置缓存中数据是否只读。只读的缓存会给所有调用者返回缓存对象的相同实例,因此这些对象不能被修改,这提供了很重要的性能优势。但读写的缓存会返回缓存对象的拷贝。这会慢一些,但是安全,因此默认是false。
size:二级缓存中可以存放的最多对象个数。默认为1024个。
2.二级缓存的存在性证明
对于映射文件中的同一个查询,肯定是同一个namespace中的查询。在一次查询后,将Sqlsession关闭,再进行一次相同查询,发现并没有到DB中进行select查询,说明二级缓存是存在的。
//证明二级缓存的存在 @Test public void test01(){ //第一次查询 Student student = dao.selectStudentById(2); System.out.println(student); sqlSession.close(); sqlSession = MyBatisUtils.getSqlSession(); dao = sqlSession.getMapper(IStudentDao.class); //第二次查询 Student student2 = dao.selectStudentById(2); System.out.println(student2); }
查看控制台:
Cache Hit Ratio表示缓存命中率。开启二级缓存后,每执行一次查询,系统都会计算一次二级缓存的命中率。第一次查询也是先从缓存中查询,只不过缓存中一定是没有的。所以会再从DB中查询。由于二级缓存中不存在该数据,所以命中率为0.但第二次查询是从二级缓存中读取的,所以这一次的命中率为1/2=0.5。当然,若有第三次查询,则命中率为1/3=0.66
3.增删改对二级缓存的影响
增删改操作,无论是否进行提交sqlSession.commit(),均会清空一级、二级缓存,使查询再次从DB中select。
测试类:
@Testpublic void test02(){ //第一次查询 Student student = dao.selectStudentById(2); System.out.println(student); sqlSession.close(); sqlSession = MyBatisUtils.getSqlSession(); dao = sqlSession.getMapper(IStudentDao.class); //插入 dao.insertStudent(new Student("",0,0)); //第二次查询 Student student2 = dao.selectStudentById(2); System.out.println(student2); }
控制台:
注意,在第二次查询时的缓存命中率为0.5,但还是从DB中查询了。说明在缓存中与该查询相对应的key是存在的,但其value被清空。而value被清空的原因是前面执行了对DB的增删改操作,所以不会从缓存中直接将null值返回,而是从DB中进行查询。
说明:
二级缓存的清空,实质上是对所查找key对应的value置为null,而非将e6fd748e50e82e7bde0d1e0efd3285df对,即entry对象删除。
从DB中进行select查询的条件是:缓存中根本不存在这个key或者缓存中存在该key所对应的entry对象,但value为null。
设置增删改操作不刷新二级缓存:
若要使某个增、删或改操作不清空二级缓存,则需要在其2fe416ebc99307b5baa877beb4126ed0或f353c4e6f3e13c8cf4dcf649d0662b12或89f2e2a6c0b1bd54a34bdf16f85e8819中添加属性flushCache="false",默认为true。
<insert id="insertStudent" flushCache="false"> insert into student(name,age,score) values(#{name},#{age},#{score}) </insert>
4.二级缓存的关闭
二级缓存默认为开启状态。若要将其关闭,则需要进行相关设置。
根据关闭的范围大小,可以分为全局关闭和局部关闭
1.全局关闭(在配置文件中设置)
全局关闭是指整个应用的二级缓存全部关闭,所有查询均不使用二级缓存。全局开关设置在主配置文件的全局设置c5c29794bda4e81a53ad53fcf163f82c中,该属性为cacheEnabled,设置为false,则关闭;设置为true,则开启,默认值为true。即二级缓存默认是开启的。
<!-- 关闭二级缓存 --> <settings> <setting name="cacheEnabled" value="false"/> </settings>
2.局部关闭(在映射文件的每个select中设置)
局部关闭指整个应用的二级缓存是开启的,但只是针对某个815c08aea17433a6272b681dc1d34168查询,不使用二级缓存。此时可以单独只关闭该815c08aea17433a6272b681dc1d34168标签的二级缓存。
在该要关闭二级缓存的815c08aea17433a6272b681dc1d34168标签中,将其属性useCache设置为false,即可关闭该查询的二级缓存。该属性默认为true,即每个815c08aea17433a6272b681dc1d34168查询的二级缓存默认是开启的。
<!--useCache="false"对当前sql的二级缓存的局部关闭 --> <select id=selectStudentById useCache="false" resultType="com.hcx.beans.Student"> select * from student where id=#{id} </select>
5.二级缓存的使用原则
1.只能在一个命名空间下使用二级缓存
由于二级缓存中的数据是基于namespace的,即不同namespace中的数据互不干扰。在多个namespace中若均存在对同一个表的操作,那么这多个namespace中的数据可能就会出现不一致现象。
2.在单表上使用二级缓存
如果一个表与其它表有关联关系,那么久非常有可能存在多个namespace对同一数据的操作。而不同namespace中的数据互补干扰,所以就有可能出现多个namespace中的数据不一致现象。
3.查询多于修改时使用二级缓存
在查询操作远远多于增删改操作的情况下可以使用二级缓存。因为任何增删改操作都将刷新二级缓存,对二级缓存的频繁刷新将降低系统性能。
三、ehcache二级查询缓存
MyBatis允许使用第三方缓存产品。ehCache就是其中一种。
注意:使用ehcache二级缓存,实体类无需实现序列化接口。
1.导入jar包
2.添加ehcache.xml
解压ehcache的核心jar包ehcache-core-2.6.8.jar,将其中的一个配置文件ehcache-failsafe.xml直接放到项目的src目录下,并更名为ehcache.xml
(1)d72dee856555ebc5179da8bdfded1f5b标签
指定一个文件目录,当内存空间不够,需要将二级缓存中数据写到硬盘上时,会写到这个指定目录中。其值一般为java.io.tmpdir,表示当前系统的默认文件临时目录。
当前文件系统的默认文件临时目录,可以通过System.property()方法查看:
@Test public void test(){ String path = System.getProperty("java.io.tmpdir"); System.out.println(path); }
(2)730eca52cf3beed4d554a3843e167f2e标签
3.启用ehcache缓存机制
在映射文件的mapper中的e20c50086bb6799ec27a6a73752fc01b中通过type指定缓存机制为Ehcache缓存。默认为MyBatis内置的二级缓存org.apache.ibatis.cache.impl.PerpetualCache。
<mapper namespace="com.hcx.dao.IStudentDao"> <cache type="org.mybatis.caches.ehcache.EhcacheCache"/> <select id=selectStudentById resultType="com.hcx.beans.Student"> select * from student where id=#{id} </select> </mapper>
4.ehcache在不同mapper中的个性化设置
在ehcache.xml中设置的属性值,会对该项目中所有使用ehcache缓存机制的缓存区域起作用。一个项目中可以有多个mapper,不同的mapper有不同的缓存区域。对于不同缓存区域也可进行专门针对于当前区域的个性设置,可通过指定不同mapper的62aecd17e676a41d3547c3bf97bb07b0属性值来设置。
62aecd17e676a41d3547c3bf97bb07b0属性值的优先级高于ehcache.xml中的属性值。
<mapper namespace="com.hcx.dao.IStudentDao"> <cache type="org.mybatis.caches.ehcache.EhcacheCache"/> <property name="maxElementsInMemory" value="5000"/> <property name="timeToIdleSeconds" value="240"/> </cache> </mapper>
以上是Java中MyBatis的查詢與快取的詳細內容。更多資訊請關注PHP中文網其他相關文章!