来源:亮哥168 发布时间:2019-03-30 17:39:04 阅读量:1592
* 有问题可以参加Java技术交流群:839366464
MyBatis缓存介绍
1 MyBatis 提供了一级缓存和二级缓存的支持
2 一级缓存: 基于PerpetualCache 的 HashMap本地缓存,其存储作用域为 Session,当 Session flush 或 close 之后,该Session中的所有 Cache 就将清空。
3. 二级缓存与一级缓存其机制相同,默认也是采用 PerpetualCache,HashMap存储,不同在于其存储作用域为 Mapper(Namespace)
4. 对于缓存数据更新机制,当某一个作用域(一级缓存Session/二级缓存Namespaces)的进行了 C/U/D 操作后,默认该作用域下所有 select 中的缓存将被clear。
5.支持自定义缓存
* 一级缓存
MyBatis 默认开启了一级缓存,一级缓存是在SqlSession 层面进行缓存的。即,同一个SqlSession ,多次调用同一个Mapper和同一个方法的同一个参数,只会进行一次数据库查询,然后把数据缓存到缓冲中,以后直接先从缓存中取出数据,不会直接去查数据库。
@Test
public void test12(){
SqlSession session = MyBatisUtils.getSqlSession();
// 测试一级缓存
UserMapper mapper = session.getMapper(UserMapper.class);
// 进行两次相同的查询操作
List<User> users = mapper.findUsers();
users = mapper.findUsers();
session.close();
}
不同的SqlSession对象,会再次发送到SQL到数据库去执行
@Test
public void test13(){
SqlSession session = MyBatisUtils.getSqlSession();
// 测试一级缓存
UserMapper mapper = session.getMapper(UserMapper.class);
// 不同sqlSession对象测试
List<User> users = mapper.findUsers();
session.commit();
// 获得一个新的SqlSession 对象
session = MyBatisUtils.getSqlSession();
mapper = session.getMapper(UserMapper.class);
users = mapper.findUsers();
session.close();
}
* 在CUD的时候会清除缓存
@Test
public void test36() throws Exception {
SqlSession sqlSession = MyBatisUtils.getSqlSession();
StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
List<Integer> ids=new ArrayList<Integer>();
ids.add(100001);
ids.add(100002);
ids.add(100003);
List<Student> students = mapper.getStudentsByListIds(ids);
Student student=new Student();
student.setSid(100002);
student.setSname("小黑");
student.setSsex('男');
student.setSage(22);
mapper.updateStudent(student);
sqlSession.commit();
students = mapper.getStudentsByListIds(ids);
System.out.println(students);
MyBatisUtils.close(sqlSession);
}
* 二级缓存
假如需要不同sqlSession对象也要缓存的话,需要开启二级缓存,是缓存在SqlSessionFactory层面给各个SqlSession 对象共享。默认二级缓存是不开启的,需要手动进行配置。
配置:
<cache/>
如果这样配置的话,很多其他的配置就会被默认进行,如:
映射文件所有的select 语句会被缓存
映射文件的所有的insert、update和delete语句会刷新缓存
缓存会使用默认的Least Recently Used(LRU,最近最少使用原则)的算法来回收缓存空间
根据时间表,比如No Flush Interval,(CNFI,没有刷新间隔),缓存不会以任何时间顺序来刷新
缓存会存储列表集合或对象(无论查询方法返回什么)的1024个引用
缓存会被视为是read/write(可读/可写)的缓存,意味着对象检索不是共享的,而且可以很安全的被调用者修改,不干扰其他调用者或县城所作的潜在修改
* 开启缓存的对象需要序列化
<mapper namespace="com.hx.hx02.mapper.UserMapper">
<cache/>
...
</mapper>
<cache eviction="LRU" flushInterval="100000" size="" readOnly="true"/>
各个属性意义如下:
eviction:缓存回收策略
- LRU:最少使用原则,移除最长时间不使用的对象
- FIFO:先进先出原则,按照对象进入缓存顺序进行回收
- SOFT:软引用,移除基于垃圾回收器状态和软引用规则的对象
- WEAK:弱引用,更积极的移除移除基于垃圾回收器状态和弱引用规则的对象
flushInterval:刷新时间间隔,单位为毫秒。如果不配置,那么只有在进行数据库修改操作才会被动刷新缓存区
size:引用额数目,代表缓存最多可以存储的对象个数
readOnly:是否只读,如果为true,则所有相同的sql语句返回的是同一个对象(有助于提高性能,但并发操作同一条数据时,可能不安全),如果设置为false,则相同的sql,后面访问的是cache的clone副本。
*细节
useCache 的使用:select 有权利选择要不要被缓存
<select id="findUsers" resultType="com.hx.hx02.bean.User" useCache="false">
SELECT *
FROM user;
</select>
二级缓存默认会在insert、update、delete操作后刷新缓存
@Test
public void test14(){
SqlSession session = MyBatisUtils.getSqlSession();
UserMapper mapper = session.getMapper(UserMapper.class);
List<User> users = mapper.findUsers();
User user=new User();
user.setUsername("xiao123");
user.setSex('男');
user.setPsw("123");
mapper.insertUser(user);
session.commit();
// 获得一个新的SqlSession 对象
session = MyBatisUtils.getSqlSession();
mapper = session.getMapper(UserMapper.class);
users = mapper.findUsers();
session.commit();
session.close();
}
flushCache:可以配置要不要刷新缓存
<insert id="insertUser" parameterType="com.hx.hx02.bean.User" flushCache="false">
INSERT INTO USER(username, psw, sex)
VALUES (#{username}, #{psw}, #{sex});
</insert>
<cache eviction="LRU" flushInterval="100" size="1" readOnly="true"/>
public void test37() throws Exception {
SqlSession sqlSession = MyBatisUtils.getSqlSession();
StudentMapper mapper = sqlSession.getMapper(StudentMapper.class);
List<Integer> ids=new ArrayList<Integer>();
ids.add(100001);
ids.add(100002);
ids.add(100003);
List<Student> students = mapper.getStudentsByListIds(ids);
sqlSession.commit();
sqlSession = MyBatisUtils.getSqlSession();
mapper = sqlSession.getMapper(StudentMapper.class);
HashMap<String,Object> map=new HashMap<String,Object>();
map.put("ssex","男");
students = mapper.getStudentsByParams(map);
sqlSession.commit();
sqlSession = MyBatisUtils.getSqlSession();
mapper = sqlSession.getMapper(StudentMapper.class);
map.put("ssex","男");
students = mapper.getStudentsByParams(map);
sqlSession.commit();
sqlSession = MyBatisUtils.getSqlSession();
mapper = sqlSession.getMapper(StudentMapper.class);
mapper.getStudentsByListIds(ids);
sqlSession.commit();
MyBatisUtils.close(sqlSession);
}
* 自定义缓存(扩展)
* mybatis 自带的缓存:
* public class PerpetualCache implements Cache
* 自定义LRUCache缓存
public class LruCache implements Cache {
private String id;
private ReadWriteLock lock = new ReentrantReadWriteLock();
private LinkedHashMap cache = new LinkedHashMap(16, 0.75f, true);
public LruCache() {
System.out.println("LruCache 初始化");
}
public LruCache(String id) {
System.out.println("LruCache 初始化:" + id);
this.id = id;
}
@Override
public String getId() {
return this.id;
}
@Override
public void putObject(Object key, Object value) {
System.out.println("放进缓存了....");
try {
lock.writeLock().lock();
cache.put(key, value);
} finally {
lock.writeLock().unlock();
}
}
@Override
public Object getObject(Object key) {
lock.readLock().lock();
try {
System.out.println("获得缓存:"+cache.get(key)+"缓存的大小:"+cache.size());
return cache.get(key);
} finally {
lock.readLock().unlock();
}
}
@Override
public Object removeObject(Object key) {
System.out.println("移除缓存对象:" + key);
try {
lock.writeLock().lock();
return cache.remove(key);
} finally {
lock.writeLock().unlock();
}
}
@Override
public void clear() {
System.out.println("清除缓存!");
cache.clear();
}
@Override
public int getSize() {
System.out.println("获取缓存大小!" + cache.size());
return cache.size();
}
@Override
public ReadWriteLock getReadWriteLock() {
System.out.println("获取锁对象!!!");
return lock;
}
}
<cache type="com.hx.hx02.cache.LruCache"/>
* Mybatis细节
* 配置mapper方式(包的方式)
<mappers>
<package name="com.hx.hx02.mapper"/>
</mappers>
* 配置package的方式出错的解决方案
解决方案,原来是IDEA maven项目默认不会把src下除java文件外的文件打包到classes文件夹下,需要在maven中增加配置如下
<resources>
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.xml</include>
</includes>
<!--默认是true-->
<!--<filtering>true</filtering>-->
</resource>
</resources>
* 实体bean的配置
<typeAliases>
<package name="com.hx.hx02.bean"></package>
</typeAliases>