高并发系统设计总结

设计方法

  • Scale-out(横向扩展):分而治之是一种常见的高并发系统设计方法,采用分布式部署的方式把流量分流开,让每个服务器都承担一部分并发和流量。

  • 缓存:使用缓存来提高系统的性能,就好比用“拓宽河道”的方式抵抗高并发大流量的冲击。

  • 异步:在某些场景下,未处理完成之前我们可以让请求先返回,在数据准备好之后再通知请求方,这样可以在单位时间内处理更多的请求。

如何实现高性能

高并发系统设计的三大目标:高性能、高可用、可扩展

提高性能

  1. 提高系统的处理核心数

  2. 减少单次任务的响应时间

高可用

评判标准

  • MTBF(Mean Time Between Failure)是平均故障间隔的意思,代表两次故障的间隔时间,也就是系统正常运转的平均时间

  • MTTR(Mean Time To Repair)表示故障的平均恢复时间,也可以理解为平均故障时间。

设计思路

系统设计

  • failover(故障转移):主从转移,水平节点间转移。例如zookeeper的节点间的转移以及主机宕机后的,从机选举主机算法(ZAB)协议。通用的协议有Paxos,Raft等。

  • 超时控制:调用其他系统的时候,做好超时控制,避免其他系统宕机导致当前服务阻塞。

  • 降级:服务降级,腾出系统资源。比如在大促期间,将历史订单和物流查询服务降级,腾出必要的系统资源给下单功能。

  • 限流:限制过大的流量对系统的冲击,从而拖垮系统。制定策略,响应符合策略的请求,对不符合策略的请求直接返回错误。

系统运维

  • 系统灰度发布,提前演练

分布式数据库

数据库架构

  • 主从同步

  • 读写分离

数据库设计

  • 分表分库

  • 生成数据库唯一id:发号器(基于雪花算法);设计数据库自增id的初始值和自增步距;UUID(可以生成唯一ID,但是不建议做数据的唯一ID)

缓存

缓存种类

  • 静态缓存:指将热点查询数据,例如秒杀商品的商品详情页生成静态页面,然后缓存在Nginx服务器上,减少服务器端压力。

  • 分布式缓存:例如Redis。通过集群部署,一致性hash做负载均衡,突破单机限制,实现海量数据的缓存;满足静态缓存无法缓存动态数据的问题。

  • 热点本地缓存:将热点数据缓存在后端应用服务器上,减少数据库交互,提高响应速度。例如商城首页的推荐商品,可以缓存在本地热点数据中,每隔30s重新拉取一次最新数据。响应的组件有Guava Cache等。

缓存读写策略

  • Cache Aside(旁路缓存策略):当有数据更新时,在更新完数据库后,删除缓存中的对应数据。下次请求到的时候,再去装载。

  • Read/Write Through(读穿 / 写穿)策略:这个策略的核心原则是用户只与缓存打交道,由缓存和数据库通信,写入或者读取数据。

  • Write Back(写回)策略:这个策略的核心思想是在写入数据时只写入缓存,并且把缓存块儿标记为“脏”的。而脏块儿只有被再次使用时才会将其中的数据写入到后端存储中。即存放脏数据的缓存被再次用来load数据时,先将脏数据提交

缓存高可用

通过部署多个节点,同时设计一些方案让这些节点互为备份。这样,当某个节点故障时,它的备份节点可以顶替它继续提供服务。分布式缓存的高可用方案一般有:客户端方案,中间代理层方案和服务端方案。

  • 客户端方案就是在客户端配置多个缓存的节点,通过缓存写入和读取算法策略来实现分布式,从而提高缓存的可用性。通过数据分片,将数据分配到多个缓存主机上。分片算法可以采用一致性hash算法。单个缓存节点可采用主从集群的方式来保证缓存的可用性。

  • 中间代理层方案是在应用代码和缓存节点之间增加代理层,客户端所有的写入和读取的请求都通过代理层,而代理层中会内置高可用策略,帮助提升缓存系统的高可用。

  • 服务端方案就是 Redis 2.4 版本后提出的 Redis Sentinel 方案。

缓存穿透

缓存穿透其实是指从缓存中没有查到数据,而不得不从后端系统(比如数据库)中查询的情况。

缓存穿透的解决方案

  • 回种空值:从数据库中查询到空值或者发生异常时,我们可以向缓存中回种一个空值,避免缓存穿透。但是因为空值并不是准确的业务数据,并且会占用缓存的空间,所以我们会给这个空值加一个比较短的过期时间,让空值在短时间之内能够快速过期淘汰。

  • 使用布隆过滤器:我们把集合中的每一个值按照提供的 Hash 算法算出对应的 Hash 值,然后将 Hash 值对数组长度取模后得到需要计入数组的索引值,并且将数组这个位置的值从 0 改成 1。在判断一个元素是否存在于这个集合中时,你只需要将这个元素按照相同的算法计算出索引值,如果这个位置的值为 1 就认为这个元素在集合中,否则则认为不在集合中。
    缺陷: 1,存在hash冲突,数据标志位有可能是错误的;2,不支持删除元素。
    使用建议: 1,选择多个 Hash 函数计算多个 Hash 值,这样可以减少误判的几率;2,布隆过滤器会消耗一定的内存空间,所以在使用时需要评估你的业务场景下需要多大的内存,存储的成本是否可以接受。

分布式锁

分布式锁的方式也比较简单,比方说 ID 为 1 的用户是一个热点用户,当他的用户信息缓存失效后,我们需要从数据库中重新加载数据时,先向 Memcached 中写入一个 Key 为"lock.1"的缓存项,然后去数据库里面加载数据,当数据加载完成后再把这个 Key 删掉。这时,如果另外一个线程也要请求这个用户的数据,它发现缓存中有 Key 为“lock.1”的缓存,就认为目前已经有线程在加载数据库中的值到缓存中了,它就可以重新去缓存中查询数据,不再穿透数据库了。

来源:https://www.icode9.com/content-4-769401.html

(0)

相关推荐

  • 高并发存储优化篇:诸多策略,缓存为王

    本文内容概述 缓存是什么 1.1. 存储宕机的致命代价 1.2. 数据库性能为什么会下降 1.3. 缓存的类型 一线研发最头疼的缓存问题 2.1. 缓存穿透 2.2. 缓存击穿 2.3. 缓存雪崩 2 ...

  • 什么是分布式缓存

    分布式缓存由一个服务端实现管理和控制,有多个客户端节点存储数据,可以进一步提高数据的读取速率.那么我们要读取某个数据的时候,应该选择哪个节点呢?如果挨个节点找,那效率就太低了.因此需要根据一致性哈希算 ...

  • 分布式数据之缓存技术,今天就一起来揭开其神秘面纱

    在这一讲,我将为你讲解分布式存储中"货架"的关键技术--缓存技术. 在计算机领域的各个方面,缓存都非常重要,是提升访问性能的一个重要技术.为什么这么说呢? 从单个计算机的体系结构来 ...

  • Java高并发21-AQS在共享,独占场景下的源码介绍

    一.AQS--锁的底层支持 1.AQS是什么 AQS是AbstractQueuedSychronizer的简称,即抽象同步队列的简称,这是实现同步器的重要组件,是一个抽象类,虽然在实际工作中很烧用到它 ...

  • 高并发场景下锁的使用技巧

    来源:张飞洪 https://www.cnblogs.com/jackyfei/p/12142840.html 如何确保一个方法,或者一块代码在高并发情况下,同一时间只能被一个线程执行,单体应用可以使 ...

  • 专业的在线考试答题系统,快考题,高并发人数使用流畅

    在线考试的普及,让越来越多的学校,企业,教育机构纷纷加入.在线考试系统的开发也打破了以往传统的考试模式,不受时间限制,不受地域限制.那么一个完善的在线考试系统除了以上两大优势,还有哪些"过人 ...

  • 高并发,我把握不住啊

    慎入,作者高并发搞得少(没搞过),这里面水太深,什么高并发,大流量的东西都是虚拟的,作者还太年轻,没有那个经历,把握不住.系统只有几QPS,开心快乐就行,不PK,文明PK. 我关注的大佬更新了,在干货 ...

  • 高并发场景下,到底先更新缓存还是先更新数据库?

    在大型系统中,为了减少数据库压力通常会引入缓存机制,一旦引入缓存又很容易造成缓存和数据库数据不一致,导致用户看到的是旧数据. 为了减少数据不一致的情况,更新缓存和数据库的机制显得尤为重要,接下来带领大 ...

  • Java高并发24-使用自定义锁生成一个消费模型

    一.使用自定义锁实现生成--消费模型 下面我们使用上节自定义的锁实现一个简单的生产--消费模型,代码如下: package com.ruigege.LockSourceAnalysis6; impor ...

  • Java高并发16-LongAdder类源码解析(下)

    一.复习 上次连载简单的介绍了其他函数的作用以及功能 二.完整的LongAdder类源码 package com.ruigege.AtomicOperationClass4;import java.u ...

  • 高并发请求测试

    前言 本篇是关于高并发请求的测试. 高并发请求的情况下,会发生什么事情呢?本文将在测试中为大家解开这个谜题. 环境配置 项目为SpringBoot项目,使用MyBatis作为持久层框架,依赖如下: 提 ...