【干货】乐视秒杀:每秒十万笔交易的数据架构解读

本贴最后更新于 2905 天前,其中的信息可能已经事过景迁

随着乐视硬件抢购的不断升级,乐视集团支付面临的请求压力百倍乃至千倍的暴增。作为商品购买的最后一环,保证用户快速稳定地完成支付尤为重要。所以在 2015 年 11 月,我们对整个支付系统进行了全面的架构升级,使之具备了每秒稳定处理 10 万订单的能力。为乐视生态各种形式的抢购秒杀活动提供了强有力的支撑。

一. 分库分表

在 redis,memcached 等缓存系统盛行的互联网时代,构建一个支撑每秒十万只读的系统并不复杂,无非是通过一致性哈希扩展缓存节点,水平扩展 web 服务器等。支付系统要处理每秒十万笔订单,需要的是每秒数十万的数据库更新操作(insert 加 update),这在任何一个独立数据库上都是不可能完成的任务,所以我们首先要做的是对订单表(简称 order)进行分库与分表。
在进行数据库操作时,一般都会有用户 ID(简称 uid)字段,所以我们选择以 uid 进行分库分表。
分库策略我们选择了“二叉树分库”,所谓“二叉树分库”指的是:我们在进行数据库扩容时,都是以 2 的倍数进行扩容。比如:1 台扩容到 2 台,2 台扩容到 4 台,4 台扩容到 8 台,以此类推。这种分库方式的好处是,我们在进行扩容时,只需 DBA 进行表级的数据同步,而不需要自己写脚本进行行级数据同步。
光是有分库是不够的,经过持续压力测试我们发现,在同一数据库中,对多个表进行并发更新的效率要远远大于对一个表进行并发更新,所以我们在每个分库中都将 order 表拆分成 10 份:order_0,order_1,....,order_9。
最后我们把 order 表放在了 8 个分库中(编号 1 到 8,分别对应 DB1 到 DB8),每个分库中 10 个分表(编号 0 到 9,分别对应 order_0 到 order_9),部署结构如下图所示:

1462851593368

根据 uid 计算数据库编号:
数据库编号 = (uid / 10) % 8 + 1
根据 uid 计算表编号:
表编号 = uid % 10
当 uid=9527 时,根据上面的算法,其实是把 uid 分成了两部分 952 和 7,其中 952 模 8 加 1 等于 1 为数据库编号,而 7 则为表编号。所以 uid=9527 的订单信息需要去 DB1 库中的 order_7 表查找。具体算法流程也可参见下图:

1462851617019

有了分库分表的结构与算法最后就是寻找分库分表的实现工具,目前市面上约有两种类型的分库分表工具:
1.客户端分库分表,在客户端完成分库分表操作,直连数据库
2.使用分库分表中间件,客户端连分库分表中间件,由中间件完成分
库分表操作
这两种类型的工具市面上都有,这里不一一列举,总的来看这两类工具各有利弊。客户端分库分表由于直连数据库,所以性能比使用分库分表中间件高 15% 到 20%。而使用分库分表中间件由于进行了统一的中间件管理,将分库分表操作和客户端隔离,模块划分更加清晰,便于 DBA 进行统一管理。
我们选择的是在客户端分库分表,因为我们自己开发并开源了一套数据层访问框架,它的代号叫“芒果”,芒果框架原生支持分库分表功能,并且配置起来非常简单。
芒果主页:mango.jfaster.org
芒果源码:github.com/jfaster/mango

二. 订单 ID

订单系统的 ID 必须具有全局唯一的特征,最简单的方式是利用数据库的序列,每操作一次就能获得一个全局唯一的自增 ID,如果要支持每秒处理 10 万订单,那每秒将至少需要生成 10 万个订单 ID,通过数据库生成自增 ID 显然无法完成上述要求。所以我们只能通过内存计算获得全局唯一的订单 ID。
JAVA 领域最著名的唯一 ID 应该算是 UUID 了,不过 UUID 太长而且包含字母,不适合作为订单 ID。通过反复比较与筛选,我们借鉴了 Twitter 的 Snowflake 算法,实现了全局唯一 ID。下面是订单 ID 的简化结构图:

1462851628576

上图分为 3 个部分:
1 时间戳
这里时间戳的粒度是毫秒级,生成订单 ID 时,使用 System.currentTimeMillis 作为时间戳
2 机器号
每个订单服务器都将被分配一个唯一的编号,生成订单 ID 时,直接使用该唯一编号作为机器号即可。
3 自增序号
当在同一服务器的同一毫秒中有多个生成订单 ID 的请求时,会在当前毫秒下自增此序号,下一个毫秒此序号继续从 0 开始。比如在同一服务器同一毫秒有 3 个生成订单 ID 的请求,这 3 个订单 ID 的自增序号部分将分别是 0,1,2。
上面 3 个部分组合,我们就能快速生成全局唯一的订单 ID。不过光全局唯一还不够,很多时候我们会只根据订单 ID 直接查询订单信息,这时由于没有 uid,我们不知道去哪个分库的分表中查询,遍历所有的库的所有表?这显然不行。所以我们需要将分库分表的信息添加到订单 ID 上,下面是带分库分表信息的订单 ID 简化结构图:

1462851653506

我们在生成的全局订单 ID 头部添加了分库与分表的信息,这样只根据订单 ID,我们也能快速的查询到对应的订单信息。
分库分表信息具体包含哪些内容?第一部分有讨论到,我们将订单表按 uid 维度拆分成了 8 个数据库,每个数据库 10 张表,最简单的分库分表信息只需一个长度为 2 的字符串即可存储,第 1 位存数据库编号,取值范围 1 到 8,第 2 位存表编号,取值范围 0 到 9。
还是按照第一部分根据 uid 计算数据库编号和表编号的算法,当 uid=9527 时,分库信息 =1,分表信息 =7,将他们进行组合,两位的分库分表信息即为"17"。具体算法流程参见下图:

1462851669654

上述使用表编号作为分表信息没有任何问题,但使用数据库编号作为分库信息却存在隐患,考虑未来的扩容需求,我们需要将 8 库扩容到 16 库,这时取值范围 1 到 8 的分库信息将无法支撑 1 到 16 的分库场景,分库路由将无法正确完成,我们将上诉问题简称为分库信息精度丢失。
为解决分库信息精度丢失问题,我们需要对分库信息精度进行冗余,即我们现在保存的分库信息要支持以后的扩容。这里我们假设最终我们会扩容到 64 台数据库,所以新的分库信息算法为:
分库信息 = (uid / 10) % 64 + 1
当 uid=9527 时,根据新的算法,分库信息=57,这里的 57 并不是真正数据库的编号,它冗余了最后扩展到 64 台数据库的分库信息精度。我们当前只有 8 台数据库,实际数据库编号还需根据下面的公式进行计算:
实际数据库编号 = (分库信息 - 1) % 8 + 1
当 uid=9527 时,分库信息 =57,实际数据库编号 =1,分库分表信息="577"。
由于我们选择模 64 来保存精度冗余后的分库信息,保存分库信息的长度由 1 变为了 2,最后的分库分表信息的长度为 3。具体算法流程也可参见下图:

1462851686408

如上图所示,在计算分库信息的时候采用了模 64 的方式冗余了分库信息精度,这样当我们的系统以后需要扩容到 16 库,32 库,64 库都不会再有问题。
上面的订单 ID 结构已经能很好的满足我们当前与之后的扩容需求,但考虑到业务的不确定性,我们在订单 ID 的最前方加了 1 位用于标识订单 ID 的版本,这个版本号属于冗余数据,目前并没有用到。下面是最终订单 ID 简化结构图:

1462851699663

Snowflake 算法:github.com/twitter/snowflake

三. 最终一致性

到目前为止,我们通过对 order 表 uid 维度的分库分表,实现了 order 表的超高并发写入与更新,并能通过 uid 和订单 ID 查询订单信息。但作为一个开放的集团支付系统,我们还需要通过业务线 ID(又称商户 ID,简称 bid)来查询订单信息,所以我们引入了 bid 维度的 order 表集群,将 uid 维度的 order 表集群冗余一份到 bid 维度的 order 表集群中,要根据 bid 查询订单信息时,只需查 bid 维度的 order 表集群即可。
上面的方案虽然简单,但保持两个 order 表集群的数据一致性是一件很麻烦的事情。两个表集群显然是在不同的数据库集群中,如果在写入与更新中引入强一致性的分布式事务,这无疑会大大降低系统效率,增长服务响应时间,这是我们所不能接受的,所以我们引入了消息队列进行异步数据同步,来实现数据的最终一致性。当然消息队列的各种异常也会造成数据不一致,所以我们又引入了实时监控服务,实时计算两个集群的数据差异,并进行一致性同步。
下面是简化的一致性同步图:

1462851711668

四. 数据库高可用

没有任何机器或服务能保证在线上稳定运行不出故障。比如某一时间,某一数据库主库宕机,这时我们将不能对该库进行读写操作,线上服务将受到影响。
所谓数据库高可用指的是:当数据库由于各种原因出现问题时,能实时或快速的恢复数据库服务并修补数据,从整个集群的角度看,就像没有出任何问题一样。需要注意的是,这里的恢复数据库服务并不一定是指修复原有数据库,也包括将服务切换到另外备用的数据库。
数据库高可用的主要工作是数据库恢复与数据修补,一般我们以完成这两项工作的时间长短,作为衡量高可用好坏的标准。这里有一个恶性循环的问题,数据库恢复的时间越长,不一致数据越多,数据修补的时间就会越长,整体修复的时间就会变得更长。所以数据库的快速恢复成了数据库高可用的重中之重,试想一下如果我们能在数据库出故障的 1 秒之内完成数据库恢复,修复不一致的数据和成本也会大大降低。
下图是一个最经典的主从结构:

1462851726511

上图中有 1 台 web 服务器和 3 台数据库,其中 DB1 是主库,DB2 和 DB3 是从库。我们在这里假设 web 服务器由项目组维护,而数据库服务器由 DBA 维护。
当从库 DB2 出现问题时,DBA 会通知项目组,项目组将 DB2 从 web 服务的配置列表中删除,重启 web 服务器,这样出错的节点 DB2 将不再被访问,整个数据库服务得到恢复,等 DBA 修复 DB2 时,再由项目组将 DB2 添加到 web 服务。
当主库 DB1 出现问题时,DBA 会将 DB2 切换为主库,并通知项目组,项目组使用 DB2 替换原有的主库 DB1,重启 web 服务器,这样 web 服务将使用新的主库 DB2,而 DB1 将不再被访问,整个数据库服务得到恢复,等 DBA 修复 DB1 时,再将 DB1 作为 DB2 的从库即可。
上面的经典结构有很大的弊病:不管主库或从库出现问题,都需要 DBA 和项目组协同完成数据库服务恢复,这很难做到自动化,而且恢复工程也过于缓慢。
我们认为,数据库运维应该和项目组分开,当数据库出现问题时,应由 DBA 实现统一恢复,不需要项目组操作服务,这样便于做到自动化,缩短服务恢复时间。
先来看从库高可用结构图:

1462851736677

如上图所示,web 服务器将不再直接连接从库 DB2 和 DB3,而是连接 LVS 负载均衡,由 LVS 连接从库。这样做的好处是 LVS 能自动感知从库是否可用,从库 DB2 宕机后,LVS 将不会把读数据请求再发向 DB2。同时 DBA 需要增减从库节点时,只需独立操作 LVS 即可,不再需要项目组更新配置文件,重启服务器来配合。
再来看主库高可用结构图:

1462851744785

如上图所示,web 服务器将不再直接连接主库 DB1,而是连接 KeepAlive 虚拟出的一个虚拟 ip,再将此虚拟 ip 映射到主库 DB1 上,同时添加 DB_bak 从库,实时同步 DB1 中的数据。正常情况下 web 还是在 DB1 中读写数据,当 DB1 宕机后,脚本会自动将 DB_bak 设置成主库,并将虚拟 ip 映射到 DB_bak 上,web 服务将使用健康的 DB_bak 作为主库进行读写访问。这样只需几秒的时间,就能完成主数据库服务恢复。
组合上面的结构,得到主从高可用结构图:

1462851752590

数据库高可用还包含数据修补,由于我们在操作核心数据时,都是先记录日志再执行更新,加上实现了近乎实时的快速恢复数据库服务,所以修补的数据量都不大,一个简单的恢复脚本就能快速完成数据修复。

五. 数据分级
支付系统除了最核心的支付订单表与支付流水表外,还有一些配置信息表和一些用户相关信息表。如果所有的读操作都在数据库上完成,系统性能将大打折扣,所以我们引入了数据分级机制。
我们简单的将支付系统的数据划分成了 3 级:
第 1 级:订单数据和支付流水数据;这两块数据对实时性和精确性要求很高,所以不添加任何缓存,读写操作将直接操作数据库。
第 2 级:用户相关数据;这些数据和用户相关,具有读多写少的特征,所以我们使用 redis 进行缓存。
第 3 级:支付配置信息;这些数据和用户无关,具有数据量小,频繁读,几乎不修改的特征,所以我们使用本地内存进行缓存。
使用本地内存缓存有一个数据同步问题,因为配置信息缓存在内存中,而本地内存无法感知到配置信息在数据库的修改,这样会造成数据库中数据和本地内存中数据不一致的问题。
为了解决此问题,我们开发了一个高可用的消息推送平台,当配置信息被修改时,我们可以使用推送平台,给支付系统所有的服务器推送配置文件更新消息,服务器收到消息会自动更新配置信息,并给出成功反馈。

六. 粗细管道
黑客攻击,前端重试等一些原因会造成请求量的暴涨,如果我们的服务被激增的请求给一波打死,想要重新恢复,就是一件非常痛苦和繁琐的过程。
举个简单的例子,我们目前订单的处理能力是平均 10 万下单每秒,峰值 14 万下单每秒,如果同一秒钟有 100 万个下单请求进入支付系统,毫无疑问我们的整个支付系统就会崩溃,后续源源不断的请求会让我们的服务集群根本启动不起来,唯一的办法只能是切断所有流量,重启整个集群,再慢慢导入流量。
我们在对外的 web 服务器上加一层“粗细管道”,就能很好的解决上面的问题。
下面是粗细管道简单的结构图:

1462851761961

请看上面的结构图,http 请求在进入 web 集群前,会先经过一层粗细管道。入口端是粗口,我们设置最大能支持 100 万请求每秒,多余的请求会被直接抛弃掉。出口端是细口,我们设置给 web 集群 10 万请求每秒。剩余的 90 万请求会在粗细管道中排队,等待 web 集群处理完老的请求后,才会有新的请求从管道中出来,给 web 集群处理。这样 web 集群处理的请求数每秒永远不会超过 10 万,在这个负载下,集群中的各个服务都会高校运转,整个集群也不会因为暴增的请求而停止服务。
如何实现粗细管道?nginx 商业版中已经有了支持,相关资料请搜索 nginx max_conns,需要注意的是 max_conns 是活跃连接数,具体设置除了需要确定最大 TPS 外,还需确定平均响应时间。
nginx 相关:
http://nginx.org/en/docs/http/ngx_http_upstream_module.html

作者:头条号 / DBAplus 社群
链接:http://toutiao.com/i6282460032487391746/
来源:头条号(今日头条旗下创作平台)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  • 架构

    我们平时所说的“架构”主要是指软件架构,这是有关软件整体结构与组件的抽象描述,用于指导软件系统各个方面的设计。另外还有“业务架构”、“网络架构”、“硬件架构”等细分领域。

    140 引用 • 441 回帖
  • 乐视
    1 引用 • 8 回帖
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    330 引用 • 614 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
请输入回帖内容 ...

推荐标签 标签

  • 房星科技

    房星网,我们不和没有钱的程序员谈理想,我们要让程序员又有理想又有钱。我们有雄厚的房地产行业线下资源,遍布昆明全城的 100 家门店、四千地产经纪人是我们坚实的后盾。

    6 引用 • 141 回帖 • 558 关注
  • Love2D

    Love2D 是一个开源的, 跨平台的 2D 游戏引擎。使用纯 Lua 脚本来进行游戏开发。目前支持的平台有 Windows, Mac OS X, Linux, Android 和 iOS。

    14 引用 • 53 回帖 • 513 关注
  • Netty

    Netty 是一个基于 NIO 的客户端-服务器编程框架,使用 Netty 可以让你快速、简单地开发出一个可维护、高性能的网络应用,例如实现了某种协议的客户、服务端应用。

    49 引用 • 33 回帖 • 23 关注
  • Flume

    Flume 是一套分布式的、可靠的,可用于有效地收集、聚合和搬运大量日志数据的服务架构。

    9 引用 • 6 回帖 • 594 关注
  • CentOS

    CentOS(Community Enterprise Operating System)是 Linux 发行版之一,它是来自于 Red Hat Enterprise Linux 依照开放源代码规定释出的源代码所编译而成。由于出自同样的源代码,因此有些要求高度稳定的服务器以 CentOS 替代商业版的 Red Hat Enterprise Linux 使用。两者的不同在于 CentOS 并不包含封闭源代码软件。

    238 引用 • 224 回帖
  • Thymeleaf

    Thymeleaf 是一款用于渲染 XML/XHTML/HTML5 内容的模板引擎。类似 Velocity、 FreeMarker 等,它也可以轻易的与 Spring 等 Web 框架进行集成作为 Web 应用的模板引擎。与其它模板引擎相比,Thymeleaf 最大的特点是能够直接在浏览器中打开并正确显示模板页面,而不需要启动整个 Web 应用。

    11 引用 • 19 回帖 • 318 关注
  • 反馈

    Communication channel for makers and users.

    123 引用 • 906 回帖 • 192 关注
  • 正则表达式

    正则表达式(Regular Expression)使用单个字符串来描述、匹配一系列遵循某个句法规则的字符串。

    31 引用 • 94 回帖
  • Rust

    Rust 是一门赋予每个人构建可靠且高效软件能力的语言。Rust 由 Mozilla 开发,最早发布于 2014 年 9 月。

    57 引用 • 22 回帖 • 4 关注
  • 博客

    记录并分享人生的经历。

    270 引用 • 2386 回帖
  • Postman

    Postman 是一款简单好用的 HTTP API 调试工具。

    4 引用 • 3 回帖 • 1 关注
  • IPFS

    IPFS(InterPlanetary File System,星际文件系统)是永久的、去中心化保存和共享文件的方法,这是一种内容可寻址、版本化、点对点超媒体的分布式协议。请浏览 IPFS 入门笔记了解更多细节。

    20 引用 • 245 回帖 • 229 关注
  • Solidity

    Solidity 是一种智能合约高级语言,运行在 [以太坊] 虚拟机(EVM)之上。它的语法接近于 JavaScript,是一种面向对象的语言。

    3 引用 • 18 回帖 • 349 关注
  • 脑图

    脑图又叫思维导图,是表达发散性思维的有效图形思维工具 ,它简单却又很有效,是一种实用性的思维工具。

    21 引用 • 58 回帖
  • Electron

    Electron 基于 Chromium 和 Node.js,让你可以使用 HTML、CSS 和 JavaScript 构建应用。它是一个由 GitHub 及众多贡献者组成的活跃社区共同维护的开源项目,兼容 Mac、Windows 和 Linux,它构建的应用可在这三个操作系统上面运行。

    15 引用 • 136 回帖 • 8 关注
  • 新人

    让我们欢迎这对新人。哦,不好意思说错了,让我们欢迎这位新人!
    新手上路,请谨慎驾驶!

    51 引用 • 226 回帖
  • 爬虫

    网络爬虫(Spider、Crawler),是一种按照一定的规则,自动地抓取万维网信息的程序。

    106 引用 • 275 回帖
  • 外包

    有空闲时间是接外包好呢还是学习好呢?

    26 引用 • 232 回帖 • 6 关注
  • CodeMirror
    1 引用 • 2 回帖 • 116 关注
  • OkHttp

    OkHttp 是一款 HTTP & HTTP/2 客户端库,专为 Android 和 Java 应用打造。

    16 引用 • 6 回帖 • 54 关注
  • etcd

    etcd 是一个分布式、高可用的 key-value 数据存储,专门用于在分布式系统中保存关键数据。

    5 引用 • 26 回帖 • 491 关注
  • CAP

    CAP 指的是在一个分布式系统中, Consistency(一致性)、 Availability(可用性)、Partition tolerance(分区容错性),三者不可兼得。

    11 引用 • 5 回帖 • 562 关注
  • ActiveMQ

    ActiveMQ 是 Apache 旗下的一款开源消息总线系统,它完整实现了 JMS 规范,是一个企业级的消息中间件。

    19 引用 • 13 回帖 • 628 关注
  • Google

    Google(Google Inc.,NASDAQ:GOOG)是一家美国上市公司(公有股份公司),于 1998 年 9 月 7 日以私有股份公司的形式创立,设计并管理一个互联网搜索引擎。Google 公司的总部称作“Googleplex”,它位于加利福尼亚山景城。Google 目前被公认为是全球规模最大的搜索引擎,它提供了简单易用的免费服务。不作恶(Don't be evil)是谷歌公司的一项非正式的公司口号。

    49 引用 • 192 回帖
  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    129 引用 • 793 回帖
  • Logseq

    Logseq 是一个隐私优先、开源的知识库工具。

    Logseq is a joyful, open-source outliner that works on top of local plain-text Markdown and Org-mode files. Use it to write, organize and share your thoughts, keep your to-do list, and build your own digital garden.

    4 引用 • 55 回帖 • 7 关注
  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3168 引用 • 8207 回帖 • 1 关注