1. 收益:java
2. 成本:redis
3. 使用场景:算法
1. 超时删除数据:也就是设置key的过时时间,好比经过expire命令设置的超时key,该策略数据一致性较低,但维护成本也低数据库
2. LRU/LFU/FIFO算法剔除:主要是针对当Redis的缓存数据达到设置的最大内存如何处理的策略,该策略数据一致性很低,但维护成本也低后端
3. 主动更新:在开发过程当中经过编写逻辑代码,控制数据更新,数据一致性高,但维护成本也高缓存
4. 使用状况:数据一致性要求低就使用最大内存时数据淘汰策略,若是数据一致性要求高,就将主动更新和超时删除结合使用,最大内存时数据淘汰策略保底服务器
1. 缓存粒度:以用户信息为例,在MySQL中用户表包含多个字段,经过select语句查询得到用户信息时,到底是选择缓存用户每一个字段的数据,仍是选择某几个重要字段的数据进行缓存,缓存全部字段或部分字段就是指缓存粒度,能够理解为缓存粒度就是指缓存对象的数据的完整性并发
2. 缓存粒度控制:运维
1. 缓存穿透:在实际应用中,业务层会先向缓存层发出数据请求,若是过这些请求没有命中(缓存层没有请求的数据),那么就会向关系型数据库层发出请求,从关系型数据库中取出数据回写到缓存层中,并返回给业务层,在下一次请求时就能够从缓存层返回数据;但若是数据库中也没有请求的数据,那么就会返回业务层空值,在后续业务层的请求同一个数据时,缓存层始终都没有数据,那么每次都会向数据库层请求数据,这样就形成了缓存穿透。异步
2. 产生缓存穿透的缘由:
3. 解决缓存穿透的方法:
1. 什么是无底洞问题:一般状况下,能够经过增长集群部署的机器数量来提高性能,可是在2010年,FaceBook发如今部署了3000个节点后发现性能反而降低;也就是说,集群中有更多的机器不表明有更好的性能,但随着数据量和并发处理量的提高,又必须提高集群的机器数量,这就是无底洞问题,这个问题没有好的解决办法,只能是经过在细节方面的优化处理来尽可能提升性能,好比优化IO操做、优化Redis集群中的批量命令执行等
1. 问题描述:若是缓存中部分key集中在一段时间内失效,发生大量的缓存穿透,全部的查询都落在数据库上,形成了缓存雪崩。形成这个问题的缘由除了是key失效之外,还多是缓存集群宕机
2. 解决方案:
设置缓存永远不过时:
从缓存上看,确实没有设置过时时间,这就保证了,不会出现热点key过时问题,也就是“物理”不过时。
从功能上看,把过时时间存在key对应的value里,若是发现要过时了,经过一个后台的异步线程进行缓存的构建,也就是“逻辑”过时
从实战看,这种方法对于性能很是友好,惟一不足的就是构建缓存时候,其他线程(非构建缓存的线程)可能访问的是老数据
示例伪代码
String get(final String key) { V v = redis.get(key); String value = v.getValue(); long timeout = v.getTimeout(); if (v.timeout <= System.currentTimeMillis()) { // 异步更新后台异常执行 threadPool.execute(new Runnable() { public void run() { String keyMutex = "mutex:" + key; if (redis.setnx(keyMutex, "1")) { redis.expire(keyMutex, 3 * 60); String dbValue = db.get(key); redis.set(key, dbValue); redis.delete(keyMutex); } } }); } return value; }
对应的示例伪代码
String get(String key) { //首先尝试从redis(或redis集群)中获取key对应的数据 String value = redis.get(key); //若是为null,则使用redis中的分布式锁 if (value == null) { //经过setnx方法建立分布式锁 if (redis.setnx(key_mutex, "1")) { // 设置分布式锁的过时时间,能够避免死锁 redis.expire(key_mutex, 3 * 60) value = db.get(key); //从数据库中取得数据 redis.set(key, value);//回写到缓存中 redis.delete(key_mutex);//释放锁 } else { //其余线程休息50毫秒后重试 Thread.sleep(50); get(key); } } }
1. 对于一些设置了过时时间的key,若是这些key可能会在某些时间点被超高并发地访问,是一种很是“热点”的数据。这个时候,须要考虑一个问题:缓存被“击穿”的问题,这个和缓存雪崩的区别在于这里针对某一key缓存,前者则是不少key。缓存在某个时间点过时的时候,刚好在这个时间点对这个Key有大量的并发请求过来,这些请求发现缓存过时通常都会从后端DB加载数据并回设到缓存,这个时候大并发的请求可能会瞬间把后端DB压垮。
2. 解决方案:
对应的示例伪代码
String get(String key) { //首先尝试从redis(或redis集群)中获取key对应的数据 String value = redis.get(key); //若是为null,则使用redis中的分布式锁 if (value == null) { //经过setnx方法建立分布式锁 if (redis.setnx(key_mutex, "1")) { // 设置分布式锁的过时时间,能够避免死锁 redis.expire(key_mutex, 3 * 60) value = db.get(key); //从数据库中取得数据 redis.set(key, value);//回写到缓存中 redis.delete(key_mutex);//释放锁 } else { //其余线程休息50毫秒后重试 Thread.sleep(50); get(key); } } }
"提早"使用互斥锁(mutex key):在value内部设置1个超时值(timeout1),。当从cache读取到timeout1发现它已通过期时候,立刻延长timeout1并从新设置到cache。而后再从数据库加载数据并设置到cache中。
设置缓存永远不过时:
从缓存上看,确实没有设置过时时间,这就保证了,不会出现热点key过时问题,也就是“物理”不过时。
从功能上看,把过时时间存在key对应的value里,若是发现要过时了,经过一个后台的异步线程进行缓存的构建,也就是“逻辑”过时
从实战看,这种方法对于性能很是友好,惟一不足的就是构建缓存时候,其他线程(非构建缓存的线程)可能访问的是老数据
示例伪代码
String get(final String key) { V v = redis.get(key); String value = v.getValue(); long timeout = v.getTimeout(); if (v.timeout <= System.currentTimeMillis()) { // 异步更新后台异常执行 threadPool.execute(new Runnable() { public void run() { String keyMutex = "mutex:" + key; if (redis.setnx(keyMutex, "1")) { redis.expire(keyMutex, 3 * 60); String dbValue = db.get(key); redis.set(key, dbValue); redis.delete(keyMutex); } } }); } return value; }