问题?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,配置文件内容不需要修改即可:
四、总结
多多看看一级缓存和二级缓存的资料,本例的下载:点击打开链接