架构第4章 高性能缓存架构

2019-08-07  本文已影响0人  小螺丝钉cici

本文参考《极客时间》- 从0开始学架构

为什么需要缓存

复杂业务场景下,单纯依靠存储系统的性能提升不够的。
典型场景:
1)需要经过复杂运算后得出的数据,存储系统无能为力。
例如,一个论坛需要在首页展示当前有多少用户同时在线,如果使用MySQL来存储当前用户状态,则每次获取这个总数都要“count(*)”大量数据,这样的操作无论怎么优化MySQL,性能都不会太高。如果要实时展示用户同时在线数,则MySQL性能无法支撑。

2)读多写少的数据,存储系统有心无力
绝大部分在线业务都是读多写少。例如,微博、淘宝、微信这类互联网业务,读业务占了整体业务量的90%以上。
以微博为例:一个明星发一条微博,可能几千万人来浏览。如果使用MySQL来存储微博,用户写微博只有一条insert语句,但每个用户浏览时都要select一次,即使有索引,几千万条select语句对MySQL数据库的压力也会非常大。

image.png

缓存的架构设计要点

缓存穿透

缓存穿透是指缓存没有发挥作用。查询缓存时,缓存中没有数据。需要去存储系统查询。
有两种情况:
1.存储数据不存在,被访问的数据确实不存在。缓存此时没有分担查询存储系统的访问压力
通常,业务上读取不存在的数据的请求量不会太大,但如果出现一些异常情况,例如被黑客攻击,故意大量访问某些读取不存在数据的业务,有可能会将存储系统拖垮。

2.缓存数据生成耗费大量时间或者资源。
存储系统有数据,但生成缓存耗费较长时间或者耗费大量资源。如果刚好在业务访问的时候缓存失效了,那么也会出现缓存没有发挥作用,访问压力全部集中在存储系统上的情况。

缓存雪崩

缓存雪崩是指当缓存失效(过期)后引起系统性能急剧下降的情况。

当缓存过期/清除后,业务系统需要重新生成缓存,再次访问存储系统,再次进行运算,这个处理步骤耗时几十毫秒甚至上百毫秒。
对于一个高并发的业务系统来说,几百毫秒内可能会接到几百上千个请求。
由于旧缓存被清除,新的缓存还未生成,并且处理这些请求线程不知道另外有一个线程正在生成缓存,因此所有的请求都会重新生成缓存,去访问存储系统,从而对存储系统造成巨大的性能压力。这些压力会拖慢整个系统,严重的会造成数据库宕机,从而形成一系列连锁反应,造成整个系统崩溃。

缓存雪崩的常见解决方法有两种:更新锁机制和后台更新机制。

后台更新既适应单机多线程的场景,也适合分布式集群的场景,相比更新锁机制要简单一些。
后台更新机制还适合业务刚上线的时候进行缓存预热。缓存预热指系统上线后,将相关的缓存数据直接加载到缓存系统,而不是等待用户访问才来触发缓存加载。

缓存热点

对于一些特别热点的数据(如电商某秒杀页面/首页banner等)
如果大部分甚至所有的业务请求都命中同一份缓存数据,则这份数据所在的缓存服务器的压力也很大。
例如,赵丽颖微博宣布“结婚,短时间内上千万的用户都会来围观,造成微博短时间挂掉

好的缓存方案可以从这几个方面入手设计:
1.什么数据应该缓存
2.什么时机触发缓存和以及触发方式是什么
3.缓存的层次和粒度( 网关缓存如 nginx,本地缓存如单机文件,分布式缓存如redis cluster,进程内缓存如全局变量)
4.缓存的命名规则和失效规则
5.缓存的监控指标和故障应对方案
6.可视化缓存数据如 redis 具体 key 内容和大小

案例:缓存持久化
1.最早采用后台用数据库,前台用关系型数据库+被动缓存的模式。结果经常性能抖动,且缓存一致性问题难解决。
后来我们的多数系统,都采用了前后台分离的模式——后台原始数据仍然是关系型数据库,前台使用缓存作为数据源。
两者之间数据实时同步+定时同步+人工触发结合。
这个模式,基本根除了穿透,雪崩,不一致,性能抖动这些。但带来了新的问题,比如数据丢失且不可恢复。
我们的做法是,让缓存具备相对可靠的持久化机制+运维体系。

2、遇到过几次热点问题,这个更棘手。
第一种情况,单Key数据结构本身过大,单个分片出现热点,单次访问的复杂度变大。
这个相对容易,可以对key进行拆分,使用hashtag机制 分片。
第二种情况,数据分片普遍不均衡,较少遇到,遇到就比较棘手。
第三种情况,数据分片均衡,但访问不均衡,可以增加副本数量。

问题一:商品列表中,像商品描述等信息,缓存更新不及时影响不大,但某些重要数据,如价格,需要及时更新的数据,有没什么好的办法做到刷新。对于价格这种关键数据,不缓存,直接从数据库查询,是否可行。或者在用户查看商品详情时再去数据库查询价格,但可能出现列表中的价格和详情页中的价格不一致。

通常有几种做法:

  1. 同步刷新缓存:当更新了某些信息后,立刻让缓存失效。 这种做法的优点是用户体验好,缺点是修改一个数据可能需要让很多缓存失效
  2. 适当容忍不一致:例如某东的商品就是这样,我查询的时候显示有货,下单的时候提示我没货了
  3. 关键信息不缓存:库存,价格等不缓存,因为这类信息查询简单,效率高,关系数据库查询性能也很高

问题二:数据库自身不是有缓存吗?
以mysql为例子:

  1. mysql第一种缓存叫sql语句结果缓存,但条件比较苛刻,程序员不可控,我们的dba线上都关闭这个功能
  2. mysql第二种缓存是innodb bufer pool,缓存的是磁盘上的分页数据,不是sql的查询结果,sql的执行过程省不了。
    而mc,redis这些实际上都是缓存sql的结果,两种缓存方式,性能差很远。
    因此,可控性,性能是数据库缓存和独立缓存的主要区别。
上一篇 下一篇

猜你喜欢

热点阅读