首頁 >Java >java教程 >Java中MyBatis的查詢與快取

Java中MyBatis的查詢與快取

巴扎黑
巴扎黑原創
2017-08-09 17:52:491674瀏覽

查詢快取的使用,主要是為了提高查詢存取速度。這篇文章主要介紹了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中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn