第一次做这种javaweb的项目,难免还是要犯很多错误。
大概也知道,redis常常被用来做应用和mysql之间的缓存。模型大概是这样子的。

为了让redis能够缓存mysql数据库中的数据,我写了很多这样类似的代码:

原来的查询商品

public Product selectProductById(int id) {Product product = productMapper.selectByPrimaryKey(id);if (product != null) {String detail = product.getDetail();if (detail != null) {product.setDetail(HtmlUtils.string2Html(detail));// 进行html转义,替换html转义符}}return product;
}

用redis缓存之后的查询商品

public Product selectProductById(int id) {Product product = JSONObject.parseObject(redisCli.get(PRODUCT_KEY +  id), Product.class);if (product != null) {product = productMapper.selectByPrimaryKey(id);String detail = product.getDetail();if (detail != null) {product.setDetail(HtmlUtils.string2Html(detail));// 进行html转义,替换html转义符}redisCli.set(PRODUCT_KEY + product.getId(), JSONObject.toJSON(product).toString(),30);}return product;
}

老板说,不行啊,网站首页太慢了!于是我们又开始在ModelAndView上做文章。
原来首页的代码

@RequestMapping("/wxIndex/{id}")
public ModelAndView goWxIndex(HttpServletRequest request, HttpServletResponse response,@PathVariable(value = "id") Integer id) {ModelAndView mv = new ModelAndView(); mv.setViewName(ViewNameConstant.WXINDEX); //一些逻辑代码return mv;
}

于是我们又加了这样的代码:

@RequestMapping("/wxIndex/{id}")
public ModelAndView goWxIndex(HttpServletRequest request, HttpServletResponse response,@PathVariable(value = "id") Integer id) {ModelAndView mv = JSONObject.parseObject(redisCli.get("index"),ModelAndView.class);if(mv != null){return mv;}mv = new ModelAndView();mv.setViewName(ViewNameConstant.WXINDEX); //一些逻辑代码redisCli.put("index",JSONObject.toString(mv),30);return mv;
}

于是代码越来越乱。

慢慢学习和适应spring的思想中,明白,我们可以使用拦截的方式去做mysql的缓存。我们拦截到一个sql语句,于是把这条sql语句作为key,把返回的结果作为value保存到redis里面去,失效时间为30秒钟;
期间如果发现一个有insert或者update就把对应表的所有的缓存给清理掉。
有了思想就下手去做好了。不曾想发现mybatis已经提供了对应好的缓存的接口Cache,思想和上述完全一致。
那么我们也就是用他的接口好了。

mybatis默认缓存是PerpetualCache,可以查看一下它的源码,发现其是Cache接口的实现;那么我们的缓存只要实现该接口即可。

该接口有以下方法需要实现:

public abstract interface CacheString getId();int getSize();void putObject(Object key, Object value);   Object getObject(Object key);                   Object removeObject(Object key);void clear();ReadWriteLock getReadWriteLock();
}

最重要的两个接口是putObject和getObject;任何select语句都会首先请求getObject函数,如果返回为null,那么再去请求mysql数据库;我们在mysql中取到数据之后,调用putObject函数,进行缓存数据的保存。
序列图为:

网上提供的案例,大部分是这样子:

public class MybatisRedisCache implements Cache {private RedisCli redisCli;@Override  public void putObject(Object key, Object value) {  logger.debug(">>>>>>>>>>>>>>>>>>>>>>>>putObject:"+key+"="+value);  redisCli.set(SerializeUtil.serialize(key.toString()), SerializeUtil.serialize(value));  }  @Override  public Object getObject(Object key) {  Object value = SerializeUtil.unserialize(redisCli.get(SerializeUtil.serialize(key.toString())));  logger.debug(">>>>>>>>>>>>>>>>>>>>>>>>getObject:"+key+"="+value);  return value;  }
}public class SerializeUtil {  public static byte[] serialize(Object object) {  ObjectOutputStream oos = null;  ByteArrayOutputStream baos = null;  try {  //序列化  baos = new ByteArrayOutputStream();  oos = new ObjectOutputStream(baos);  oos.writeObject(object);  byte[] bytes = baos.toByteArray();  return bytes;  } catch (Exception e) {  e.printStackTrace();  }  return null;  }  public static Object unserialize(byte[] bytes) {  ByteArrayInputStream bais = null;  try {  //反序列化  bais = new ByteArrayInputStream(bytes);  ObjectInputStream ois = new ObjectInputStream(bais);  return ois.readObject();  } catch (Exception e) {  }  return null;  }
} 

如果是通过java提供的序列化进行实体类和String的转换,那么我们要修改所有已经存在的实体Bean类,工作量太大;而且java的序列化效率又低;我们还是考虑使用工程已经引入的fastjson好;使用fastjson,就必须在缓存数据的时候,同时缓存数据的类型;我们使用redis的hash结构,就能解决这个问题

于是接口就成了下面这个样子:

public class MybatisRedisCache implements Cache {private RedisCli redisCli;@Overridepublic void putObject(Object key, Object value) {String keyStr = getKey(key);Map<String,String> map = new HashMap<String,String>();//如果是多组数据,那么保存的方式不同,多组的情况需要保存子实体类型if(value.getClass().equals(ArrayList.class)){@SuppressWarnings("unchecked")List<Object> list = (List<Object>)value;map.put("type", "java.util.ArrayList");if(list.size() > 0){map.put("subType", list.get(0).getClass().getCanonicalName());}else{map.put("subType",Object.class.getCanonicalName());}map.put("value", JSONObject.toJSONString(value));}else{map.put("type", value.getClass().getCanonicalName());map.put("value", JSONObject.toJSONString(value));}this.redisCli.hAllSet(keyStr, map,30);this.cacheKeys.add(keyStr);}@Overridepublic Object getObject(Object key) {try{String keyStr = getKey(key);Map<Object,Object> map = this.redisCli.hAllGet(keyStr);String type = (String)map.get("type");String value = (String)map.get("value");if(type == null || value == null){return null;}if("java.util.ArrayList".equals(type)){String subType = (String)map.get("subType");return JSONObject.parseArray(value, Class.forName(subType));}else{return JSONObject.parseObject(value, Class.forName(type));}}catch (Exception e){e.printStackTrace();return null;}}@Overridepublic void clear() {if(this.cacheKeys.isEmpty()){return ;} for(String key : this.cacheKeys){this.redisCli.del(key);}this.cacheKeys.clear();}
}

ps: 我们这里还是把key直接保存在了内存里面,这样存在的问题就是,如果服务器重启,那么需要清理所有的缓存;不然一定会造成脏数据。
或者,我们在保存缓存数据的时候,设置缓存数据的生命时间是30秒即可,希望对大家有所帮助。

转载于:https://www.cnblogs.com/archy_yu/p/5276153.html

使用redis和fastjson做应用和mysql之间的缓存相关推荐

  1. redis 和 数据库mysql之间的关系

    https://www.zhihu.com/question/20734566 https://www.zhihu.com/question/19660689 http://blog.csdn.net ...

  2. redis 如何 mysql_Redis 如何保持和 MySQL 数据一致

    一.需求起因 在高并发的业务场景下,数据库大多数情况都是用户并发访问最薄弱的环节.所以,就需要使用redis做一个缓冲操作,让请求先访问到redis,而不是直接访问MySQL等数据库. 这个业务场景, ...

  3. 放弃redis使用mongodb做任务队列支持增删改管理

    使用mongodb做任务队列管理支持增删改 ,我是喜欢用redis的list做队列的,但是这边经常堆积任务,需要查看并删除清空队列,有可能会根据一些content的字段来进行删除,有可能是会针对发件人 ...

  4. MySQL redis如何实现_如何保障mysql和redis之间的数据一致性?(转发)

    在高并发的业务场景下,数据库大多数情况都是用户并发访问最薄弱的环节.所以,就需要使用redis做一个缓冲操作,让请求先访问到redis,而不是直接访问Mysql等数据库.这样可以大大缓解数据库的压力. ...

  5. php redis都可以做什么,redis除了缓存还能做什么

    Redis应该说是目前最受欢迎的NoSQL数据库之一了.Redis通常被作为缓存组件,用作缓存数据.不过,除了可以缓存数据,其实Redis可以做的事还有很多. 下面列举几例,供大家参考.推荐:< ...

  6. redis夺命连环问10--说说Redis是怎么做旁路缓存的?

    目录 相关前置知识文章 说说Redis是怎么做旁路缓存的? 先谈缓存大概怎么做 再谈旁路缓存两种模式 redis和mysql如何保证数据一致性? 那怎么解决缓存和数据库的数据不一致问题? 如何保证缓存 ...

  7. Redis是如何做缓存的

    前言 基于Redis的高性能特性,我们将Redis用在缓存场景非常广泛.使用起来方便,响应也是远超关系型数据库. 但我们用Redis做缓存时,也要注意各种问题的应对和措施,比如缓存失效.数据一致性问题 ...

  8. 工作笔记-使用Redis作为Mysql数据库的缓存

    有一个新需求,需要使用Redis做为Mysql的缓存,需要做全表缓存. Redis是K/V的简单键值对存储形式,MySQL是传统关系型数据库,是一维和二维的区别,怎么把表映射到Redis中呢. Red ...

  9. redis cluster 设置密码做集群时gem下client.rb文件修改

    redis cluster 设置密码做集群时gem下client.rb文件修改 来源 https://www.cnblogs.com/shihaiming/p/5949772.html redis节点 ...

最新文章

  1. CV_IMAGE_ELEM参数赋值时注意的问题
  2. MySQL数据库root账户密码忘记两种处理方法转载
  3. html+not选择器,CSS3属性选择器与(:not)选择器_html/css_WEB-ITnose
  4. 神策学堂发布——你有一份数据驱动指南待查收
  5. MATLAB | matlab运行、下载链接及21个matlab基本图像调试代码
  6. 直播未来属于RTMP还是HTTP?
  7. 跟我一起学WCF(2)——利用.NET Remoting技术开发分布式应用
  8. Bitmap详解(中)之像素级操作
  9. hashmap 允许key重复吗_HashTable和HashMap的区别详解
  10. 推荐算法工程师必备!!!协同过滤推荐算法总结
  11. Linux 搭建NTP服务器
  12. 微型计算机系统的几个指标课件,[计算机基础课件67.ppt
  13. LIF-Seg 激光雷达与相机图像融合的三维激光雷达语义分割
  14. 如何快速填充表格公式
  15. mediawiki搭建使用
  16. Winform 连接打印机
  17. 再见,2017,你好,2018
  18. matlab实现扫雷小游戏
  19. oneDNS解决google等登陆问题
  20. python语言的变量_自兴人工智能------Python语言的变量认识及操作

热门文章

  1. 事件响应政策制定常见雷区,都踩了我就只能祝福你了……
  2. linux 查看端口被哪个程序占用
  3. Win7安装visual c++ 2015 redistributable x64失败
  4. maven 使用 问题记录
  5. Java字符串的10大热点问题盘点
  6. 【2012年华为校园招聘软开上机-成都】字母转换、统计单词个数
  7. 使用图形工具管理Server Core上的账号和组
  8. ValueError: invalid \x escape
  9. SQL Server LOWER Functions
  10. 如果检测指定的Windows服务是否启动