redis作为现在最优秀的key-value数据库,非常适合提供项目的缓存服务。把redis作为mybatis的查询缓存也是很常见的做法。在网上发现N多人是自己做的Cache,其实在mybatis的git下有一个子项目mybatis-redis;这个项目提供了redis作为mybatis查询缓存的一个实现,下面先分析一下这个项目的实现原理,再提出几个项目的问题:

代码实现

该项目和大家普遍实现Mybatis的缓存方案大同小异,无非是实现Cache接口,并使用jedis操作缓存;不过该项目在设计细节上有一些区别;下面简要分析一下源码:

  1. public final class RedisCache implements Cache {
  2. public RedisCache(final String id) {
  3. if (id == null) {
  4. throw new IllegalArgumentException("Cache instances require an ID");
  5. }
  6. this.id = id;
  7. RedisConfig redisConfig = RedisConfigurationBuilder.getInstance().parseConfiguration();
  8. pool = new JedisPool(redisConfig, redisConfig.getHost(), redisConfig.getPort(),
  9. redisConfig.getConnectionTimeout(), redisConfig.getSoTimeout(), redisConfig.getPassword(),
  10. redisConfig.getDatabase(), redisConfig.getClientName());
  11. }

RedisCache在mybatis启动的时候,由MyBatis的CacheBuilder创建,创建的方式很简单,就是调用RedisCache的带有String参数的构造方法,即RedisCache(String id);而在RedisCache的构造方法中,调用了RedisConfigurationBuilder来创建RedisConfig对象,并使用RedisConfig来创建JedisPool。

RedisConfig类继承了JedisPoolConfig,并提供了host,port等属性的包装,简单看一下RedisConfig的属性:

  1. public class RedisConfig extends JedisPoolConfig {
  2. private String host = Protocol.DEFAULT_HOST;
  3. private int port = Protocol.DEFAULT_PORT;
  4. private int connectionTimeout = Protocol.DEFAULT_TIMEOUT;
  5. private int soTimeout = Protocol.DEFAULT_TIMEOUT;
  6. private String password;
  7. private int database = Protocol.DEFAULT_DATABASE;
  8. private String clientName;
  9. }

RedisConfig对象是由RedisConfigurationBuilder创建的,简单看下这个类的主要方法:

  1. public RedisConfig parseConfiguration(ClassLoader classLoader) {
  2. Properties config = new Properties();
  3. InputStream input = classLoader.getResourceAsStream(redisPropertiesFilename);
  4. if (input != null) {
  5. try {
  6. config.load(input);
  7. } catch (IOException e) {
  8. throw new RuntimeException(
  9. "An error occurred while reading classpath property '"
  10. + redisPropertiesFilename
  11. + "', see nested exceptions", e);
  12. } finally {
  13. try {
  14. input.close();
  15. } catch (IOException e) {
  16. // close quietly
  17. }
  18. }
  19. }
  20. RedisConfig jedisConfig = new RedisConfig();
  21. setConfigProperties(config, jedisConfig);
  22. return jedisConfig;
  23. }

核心的方法就是parseConfiguration方法,该方法从classpath中读取一个redis.properties文件:

  1. host=localhost
  2. port=6379
  3. connectionTimeout=5000
  4. soTimeout=5000
  5. password=
  6. database=0
  7. clientName=

并将该配置文件中的内容设置到RedisConfig对象中,并返回;

接下来,就是RedisCache使用RedisConfig类创建完成JedisPool;

在RedisCache中实现了一个简单的模板方法,用来操作Redis:

  1. private Object execute(RedisCallback callback) {
  2. Jedis jedis = pool.getResource();
  3. try {
  4. return callback.doWithRedis(jedis);
  5. } finally {
  6. jedis.close();
  7. }
  8. }

模板接口为RedisCallback,这个接口中就只需要实现了一个doWithRedis方法而已:

  1. public interface RedisCallback {
  2. Object doWithRedis(Jedis jedis);
  3. }

接下来看看Cache中最重要的两个方法:putObject和getObject,通过这两个方法来查看mybatis-redis储存数据的格式:

  1. @Override
  2. public void putObject(final Object key, final Object value) {
  3. execute(new RedisCallback() {
  4. @Override
  5. public Object doWithRedis(Jedis jedis) {
  6. jedis.hset(id.toString().getBytes(), key.toString().getBytes(), SerializeUtil.serialize(value));
  7. return null;
  8. }
  9. });
  10. }
  11. @Override
  12. public Object getObject(final Object key) {
  13. return execute(new RedisCallback() {
  14. @Override
  15. public Object doWithRedis(Jedis jedis) {
  16. return SerializeUtil.unserialize(jedis.hget(id.toString().getBytes(), key.toString().getBytes()));
  17. }
  18. });
  19. }

可以很清楚的看到,mybatis-redis在存储数据的时候,是使用的hash结构,把cache的id作为这个hash的key(cache的id在mybatis中就是mapper的namespace);这个mapper中的查询缓存数据作为hash的field,需要缓存的内容直接使用SerializeUtil存储,SerializeUtil和其他的序列化类差不多,负责对象的序列化和反序列化;

使用方式

整个mybatis-redis的关键代码就这些,通过对代码的分析,我们很容易得到mybatis-redis的使用方式:

1,在项目中添加一个redis.properties配置;

2,直接在mapper中使用<cache type="org.mybatis.caches.redis.RedisCache" />即可;

分析

通过代码,我们可以看到在实际应用当中可能存在的一些问题:

  1. 默认情况下,mybatis会为每一个mapper创建一个RedisCache,而JedisPool是在RedisCache的构造方法中创建的,这就意味着会为每一个mapper创建一个JedisPool,使用意图和开销上面都会有问题;
  2. 在很多情况下,我们的应用中也会独立使用到redis,这样也无法让RedisCache直接使用我们项目中可能已经存在的JedisPool;并且会造成两个配置文件(除非我们应用也使用redis.properties);
  3. RedisCache是使用hash来缓存一个Mapper中的查询,所以我们只能通过mybatis的cache配置来控制对象的生存时间,空闲时间等属性;而无法独立的去配置每一个缓存区域(即每一个hash);
文/小码哥Java学院(简书作者)

原文链接:http://www.jianshu.com/p/648c10420df1

著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

mybatis-redis项目分析的更多相关文章

  1. SpringMVC + Spring + Mybatis+ Redis +shiro以及MyBatis学习

    SpringMVC + Spring + Mybatis+ Redis +shiro http://www.sojson.com/shiro MyBatis简介与配置MyBatis+Spring+My ...

  2. Spring Boot + Mybatis + Redis二级缓存开发指南

    Spring Boot + Mybatis + Redis二级缓存开发指南 背景 Spring-Boot因其提供了各种开箱即用的插件,使得它成为了当今最为主流的Java Web开发框架之一.Mybat ...

  3. springboot整合mybatis,redis,代码(二)

    一 说明: springboot整合mybatis,redis,代码(一) 这个开发代码的复制粘贴,可以让一些初学者直接拿过去使用,且没有什么bug 二 对上篇的说明 可以查看上图中文件: 整个工程包 ...

  4. 如何玩转最新的项目的搭配springmvc+mybatis+Redis+Nginx+tomcat+mysql

    上一次完成nginx+tomcat组合搭配,今天我们就说说,这几个软件在项目中充当的角色: 要想完成这几个软件的组合,我们必须知道和熟悉应用这个框架, 一: Nginx:在项目中大多数作为反向代理服务 ...

  5. SpringCloud+MyBatis+Redis整合—— 超详细实例(二)

    2.SpringCloud+MyBatis+Redis redis①是一种nosql数据库,以键值对<key,value>的形式存储数据,其速度相比于MySQL之类的数据库,相当于内存读写 ...

  6. springmvc+spring-security+mybatis +redis +solar框架抽取

    参考文章:Spring MVC 3 深入总结: 第二章 Spring MVC入门 —— 跟开涛学SpringMVC 参考博客:http://www.cnblogs.com/liukemng/categ ...

  7. springmvc+mybatis+redis(转)

    最近在学习redis的使用方法,它的本地使用方法比较简单,只需要先启动Redis服务器,然后运行测试代码即可.但是现在我想要在网站上访问数据库的时候采用Redis缓存,问题就出来了.要么是缓存直接失效 ...

  8. Springboot Mybatis Redis 实现二级缓存

    前言 什么是mybatis二级缓存? 二级缓存是多个sqlsession共享的,其作用域是mapper的同一个namespace. 即,在不同的sqlsession中,相同的namespace下,相同 ...

  9. springboot mybatis redis 二级缓存

    前言 什么是mybatis二级缓存? 二级缓存是多个sqlsession共享的,其作用域是mapper的同一个namespace. 即,在不同的sqlsession中,相同的namespace下,相同 ...

  10. springboot+mybatis+redis实现分布式缓存

    大家都知道springboot项目都是微服务部署,A服务和B服务分开部署,那么它们如何更新或者获取共有模块的缓存数据,或者给A服务做分布式集群负载,如何确保A服务的所有集群都能同步公共模块的缓存数据, ...

随机推荐

  1. 自定义Chrome 背景色 (FF需要User Styles插件)

    Chrome有个自定义背景色的文件  Custom.css 默认里面什么字都没写 html, body {background-color: #e0dcc0!important;}      这个颜色 ...

  2. Net Core WebAPI

    Net Core WebAPI .Net Core WebAPI 基于Task的同步&异步编程快速入门 Task.Result async & await 总结 并行任务(Task)以 ...

  3. 禁用Java DNS缓存-Disable DNS caching

    Once an application has performed network access (i.e. urlconnection, parsing of xml document with e ...

  4. Struts2 中action之间的跳转(分享)

    例如从你的login.action到register.action  有两种实现方式 1. 设置type="redirect" <package  name="st ...

  5. Windows Azure 网站自愈

    编辑人员注释:本文章由 Windows Azure 网站团队的项目经理Apurva Joshi 撰写. 您有多少次在半夜被叫醒去解决一个仅需重新启动网站即可解决的问题?要是可以自动检测一些状况并自动恢 ...

  6. JAVA GUI学习 - 窗口【x】按钮关闭事件触发器:重写processWindowEvent(WindowEvent e)方法

    public class WindowListenerKnow extends JFrame { public WindowListenerKnow() { this.setBounds(300, 1 ...

  7. Android中获取系统的时间

    activity代码 public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); set ...

  8. 具体解释VMware 9.0.1安装MAC OS X 10.8(历时近3日感想篇)

    突然心血来潮,想用VMware 9.0.1安装MAC OS X,但网上的文章多多少少总有点缺陷,不能适合每个人,在综合了近30篇安装MAC OS X的文章后,我决定公布一篇比較大众化,比較详尽的MAC ...

  9. levelDB缓存实现

    leveldb的缓存机制 leveldb采用LRU机制, 利用键的哈希值前n位作为索引, 将要插入的键值对分派到指定的缓存区, 当缓存区的使用率大于总容量后, 优先淘汰最近最少使用的缓存, 独立的缓存 ...

  10. java覆写equals方法

    何时需要重写equals() 当一个类有自己特有的“逻辑相等”概念(不同于对象身份的概念). object规范规定,如果要重写equals(),也要重写hashcode() 如何覆写equals() ...