问题?mybatis之一级、二级缓存的问题详解
一、一级缓存、二级缓存的定义及区别
我们知道mybatis的对JDBC 的一个轻量级的封装,在运行速度方面优于hibernate。
一级缓存是SqlSession级别的缓存。在操作数据库时需要构造 sqlSession对象,在对象中有一个(内存区域)数据结构(HashMap)用于存储缓存数据。不同的sqlSession之间的缓存数据区域(HashMap)是互相不影响的。 一级缓存的作用域是同一个SqlSession,在第一个sqlSession执行相同的sql语句后结果放在内存中,第二次会从缓存中获取数据将不再从数据库查询,从而提高查询效率。当一个sqlSession结束后该sqlSession中的一级缓存也就不存在了。Mybatis默认开启一级缓存。
本地缓存不能被关闭,可以调用clearCache()来清空本地缓存,或者改变缓存的作用域
二级缓存是mapper级别的缓存,多个SqlSession去操作同一个Mapper的sql语句,多个SqlSession去操作数据库得到数据会存在二级缓存区域,多个SqlSession可以共用二级缓存,二级缓存是跨SqlSession的。
二级缓存是多个SqlSession共享的,其作用域是mapper的同一个namespace,第一次执行完毕会将数据库中查询的数据写到缓存(内存),第二次会从缓存中获取数据将不再从数据库查询,从而提高查询效率。
Mybatis默认没有开启二级缓存需要在setting全局参数中配置开启二级缓存。
如果缓存中有数据就不用从数据库中获取,大大提高系统性能。
二、mybatis的一级二级缓存的使用
我这里采用的是面向接口来操作Mapper.xml文件的
读取SqlSessionFactory工具类:
package com.itcast.util;
import java.io.IOException;
import java.io.InputStream;
import org.apache.ibatis.io.Resources;
import org.apache.ibatis.session.SqlSessionFactory;
import org.apache.ibatis.session.SqlSessionFactoryBuilder;
public class MyBatisSqlSessionFactiorUtil {
public static SqlSessionFactory factory;
static{
try {
String resource = "sqlMapConfig.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
factory = new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
e.printStackTrace();
}
}
}
一缓缓存:
/*一级缓存检测*/
@Test
public void test1(){
SqlSession session = factory.openSession();
//得到接口
UserMapper mapper = session.getMapper(UserMapper.class);
Map map = new HashMap();
map.put("name", "'lijun'");
List<User> list = mapper.find(map);
for(User user:list){
System.out.println(user.toString());
}
System.out.println("第二次查询直接从缓存汇总拿了,没有发出sql语句:");
UserMapper mapper2 = session.getMapper(UserMapper.class);
List<User> list2 = mapper2.find(map);
for(User user:list2){
System.out.println(user.toString());
}
}
二级缓存:
mybatis的二级缓存默认是没有开启的,要使用需要开启的。在sqlMapConfig.xml文件中的<configuration>下加上
<!-- 设置全局二级缓存开启 这个必须放在别名之上,不然会报错误,直接放在第一个 -->
<span lang="EN-US" style="font-size: 11pt; font-family: 'Courier New';"><!-- 设置全局二级缓存开启 这个必须放在别名之上,不然会报错误 -->
<span style="white-space:pre"> </span><settings>
<span style="white-space:pre"> </span><setting name="cacheEnabled" value="true"/>
<span style="white-space:pre"> </span></settings></span>
在Mapper.xml中namespace中使用缓存
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<!-- 配置命名空间,区别名称 -->
<mapper namespace="cn.itcast.mapper.PersonMapper">
<!-- 使用二级缓存 -->
<cache/>
正常情况下,这样使用就可以了。并且测试也成功:
/*二级缓存检测,默认情况下,二级缓存是没有开启的,我们需要开启*/
@Test
public void test2(){
SqlSession session = factory.openSession();
//得到接口
UserMapper mapper = session.getMapper(UserMapper.class);
Map map = new HashMap();
map.put("name", "'lijun'");
List<User> list = mapper.find(map);
for(User user:list){
System.out.println(user.toString());
}
session.close();
session = factory.openSession();
System.out.println("二级缓存:第二次查询直接从缓存汇总拿了,没有发出sql语句:");
UserMapper mapper2 = session.getMapper(UserMapper.class);
List<User> list2 = mapper2.find(map);
for(User user:list2){
System.out.println(user.toString());
}
}
成功获取二级缓存:
上面也是正常情况下,就可以了。但是二级缓存也是可以优化的。二级缓存怎么优化?
在实际开发中,经过在线上压力测试,找到一个符合当前客户运行环境下最恰当的值。
默认mybatis二级缓存一般,可以进行配置
另外还有就是使用第三方的Cache《Ehcache》
第三方cache中有两个佼佼者;可以使用第三方包,包在本例下载页
Ehcache 单机
Momeycache 集群
加入后,再配置一下
<cachetype="org.mybatis.caches.ehcache.EhcacheCache"/>
但是这样测试会报这样的错,说是缺少配置文件,比如:
将jar包中的下列配置文件拷入src下,并且更名为ehcache.xml,配置文件内容不需要修改即可:
四、总结
多多看看一级缓存和二级缓存的资料,本例的下载:点击打开链接