Files
CategoryResourceRepost/极客时间专栏/左耳听风/分布式架构/26 | 分布式系统关键技术:流量与数据调度.md
louzefeng d3828a7aee mod
2024-07-11 05:50:32 +00:00

147 lines
11 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<audio id="audio" title="26 | 分布式系统关键技术:流量与数据调度" controls="" preload="none"><source id="mp3" src="https://static001.geekbang.org/resource/audio/fd/e7/fd4a81af29ceabe9e6a12bdffd496ee7.mp3"></audio>
关于流量调度现在很多架构师都把这个事和服务治理混为一谈了。我觉得还是应该分开的。一方面服务治理是内部系统的事而流量调度可以是内部的更是外部接入层的事。另一方面服务治理是数据中心的事而流量调度要做得好应该是数据中心之外的事也就是我们常说的边缘计算是应该在类似于CDN上完成的事。
所以,流量调度和服务治理是在不同层面上的,不应该混在一起,所以在系统架构上应该把它们分开。
# 流量调度的主要功能
对于一个流量调度系统来说,其应该具有的主要功能是:
<li>
依据系统运行的情况,自动地进行流量调度,在无需人工干预的情况下,提升整个系统的稳定性;
</li>
<li>
让系统应对爆品等突发事件时,在弹性计算扩缩容的较长时间窗口内或底层资源消耗殆尽的情况下,保护系统平稳运行。
</li>
这还是为了提高系统架构的稳定性和高可用性。
此外,这个流量调度系统还可以完成以下几方面的事情。
- **服务流控**。服务发现、服务路由、服务降级、服务熔断、服务保护等。
- **流量控制**。负载均衡、流量分配、流量控制、异地灾备(多活)等。
- **流量管理**。协议转换、请求校验、数据缓存、数据计算等。
所有的这些都应该是一个API Gateway应该做的事。
# 流量调度的关键技术
但是作为一个API Gateway来说因为要调度流量首先需要扛住流量而且还需要有一些比较轻量的业务逻辑所以一个好的API Gateway需要具备以下的关键技术。
<li>
**高性能**。API Gateway必须使用高性能的技术所以也就需要使用高性能的语言。
</li>
<li>
**扛流量**。要能扛流量就需要使用集群技术。集群技术的关键点是在集群内的各个结点中共享数据。这就需要使用像Paxos、Raft、Gossip这样的通讯协议。因为Gateway需要部署在广域网上所以还需要集群的分组技术。
</li>
<li>
**业务逻辑**。API Gateway需要有简单的业务逻辑所以最好是像AWS的Lambda 服务一样,可以让人注入不同语言的简单业务逻辑。
</li>
<li>
**服务化**。一个好的API Gateway需要能够通过Admin API来不停机地管理配置变更而不是通过一个.conf文件来人肉地修改配置。
</li>
基于上述的这几个技术要求就其本质来说目前可以做成这样的API Gateway几乎没有。这也是为什么我现在自己自主开发的原因你可以到我的官网MegaEase.com上查看相关的产品和技术信息
# 状态数据调度
对于服务调度来说最难办的就是有状态的服务了。这里的状态是State也就是说有些服务会保存一些数据而这些数据是不能丢失的所以这些数据是需要随服务一起调度的。
一般来说我们会通过“转移问题”的方法来让服务变成“无状态的服务”。也就是说会把这些有状态的东西存储到第三方服务上比如Redis、MySQL、ZooKeeper或是NFS、Ceph的文件系统中。
这些“转移问题”的方式把问题转移到了第三方服务上于是自己的Java或PHP服务中没有状态但是Redis和MySQL上则有了状态。所以我们可以看到现在的分布式系统架构中出问题的基本都是这些存储状态的服务。
因为数据存储结点在Scale上比较困难所以成了一个单点的瓶颈。
# 分布式事务一致性的问题
要解决数据结点的Scale问题也就是让数据服务可以像无状态的服务一样在不同的机器上进行调度这就会涉及数据的replication问题。而数据replication则会带来数据一致性的问题进而对性能带来严重的影响。
要解决数据不丢失的问题,只能通过数据冗余的方法,就算是数据分区,每个区也需要进行数据冗余处理。这就是数据副本。当出现某个节点的数据丢失时,可以从副本读到。数据副本是分布式系统解决数据丢失异常的唯一手段。简单来说:
1. 要想让数据有高可用性,就得写多份数据。
1. 写多份会引起数据一致性的问题。
1. 数据一致性的问题又会引发性能问题。
在解决数据副本间的一致性问题时,我们有一些技术方案。
- Master-Slave方案。
- Master-Master方案。
- 两阶段和三阶段提交方案。
- Paxos方案。
你可以仔细地读一下我在3年前写的[《分布式系统的事务处理》这篇文章](https://coolshell.cn/articles/10910.html)。其中我引用了Google App Engine联合创始人赖安·巴里特Ryan Barrett在2009年Google I/O上的演讲[Transaction Across DataCenter视频](http://www.youtube.com/watch?v=srOgpXECblk) 中的一张图。
<img src="https://static001.geekbang.org/resource/image/e5/ec/e566933d9967f2f5e0f4dcddc66247ec.png" alt="" />
从上面这张经典的图中,我们可以看到各种不同方案的对比。
现在很多公司的分布式系统事务基本上都是两阶段提交的变种。比如阿里推出的TCCTryConfirmCancel或是我在亚马逊见到的PlanReserveConfirm的方式等等。凡是通过业务补偿或是在业务应用层上做的分布式事务的玩法基本上都是两阶段提交或是两阶段提交的变种。
换句话说迄今为止在应用层上解决事务问题只有“两阶段提交”这样的方式而在数据层解决事务问题Paxos算法则是不二之选。
# 数据结点的分布式方案
真正完整解决数据Scale问题的应该还是数据结点自身。只有数据结点自身解决了这个问题才能做到对上层业务层的透明业务层可以像操作单机数据库一样来操作分布式数据库这样才能做到整个分布式服务架构的调度。
也就是说这个问题应该解决在数据存储方。但是因为数据存储结果有太多不同的Scheme所以现在的数据存储也是多种多样的有文件系统有对象型的有Key-Value式有时序的有搜索型的有关系型的……
这就是为什么分布式数据存储系统比较难做,因为很难做出来一个放之四海皆准的方案。类比一下编程中的各种不同的数据结构你就会明白为什么会有这么多的数据存储方案了。
但是我们可以看到,这个“数据存储的动物园”中,基本上都在解决数据副本、数据一致性和分布式事务的问题。
比如AWS的Aurora就是改写了MySQL的InnoDB引擎。为了承诺高可用的SLA所以需要写6个副本但实现方式上它不像MySQL通过bin log的数据复制方式而是更为“惊艳”地复制SQL语句然后拼命地使用各种tricky的方式来降低latency。比如使用多线程并行、使用SQL操作的merge等。
MySQL官方也有MySQL Cluster的技术方案。此外MongoDB、国内的PingCAP的TiDB、国外的CockroachDB还有阿里的OceanBase都是为了解决大规模数据的写入和读取的问题而出现的数据库软件。所以我觉得成熟的可以用到生产线上的分布式数据库这个事估计也不远了。
而对于一些需要文件存储的则需要分布式文件系统的支持。试想一个Kafka或ZooKeeper需要把它们的数据存储到文件系统上。当这个结点有问题时我们需要再启动一个Kafka或ZooKeeper的实例那么也需要把它们持久化的数据搬迁到另一台机器上。
注意虽然Kafka和ZooKeeper是HA的数据会在不同的结点中进行复制但是我们也应该搬迁数据这样有利用于新结点的快速启动。否则新的结点需要等待数据同步这个时间会比较长可能会导致数据层的其它问题。
于是我们就需要一个底层是分布式的文件系统这样新的结点只需要做一个简单的远程文件系统的mount就可以把数据调度到另外一台机器上了。
所以真正解决数据结点调度的方案应该是底层的数据结点。在它们上面做这个事才是真正有效和优雅的。而像阿里的用于分库分表的数据库中间件TDDL或是别的公司叫什么DAL之类的这样的中间件都会成为过渡技术。
## 状态数据调度小结
接下来,我们对状态数据调度做个小小的总结。
<li>
对于应用层上的分布式事务一致性,只有两阶段提交这样的方式。
</li>
<li>
而底层存储可以解决这个问题的方式是通过一些像Paxos、Raft或是NWR这样的算法和模型来解决。
</li>
<li>
状态数据调度应该是由分布式存储系统来解决的这样会更为完美。但是因为数据存储的Scheme太多所以导致我们有各式各样的分布式存储系统有文件对象的有关系型数据库的有NoSQL的有时序数据的有搜索数据的有队列的……
</li>
总之我相信状态数据调度应该是在IaaS层的数据存储解决的问题而不是在PaaS层或者SaaS层来解决的。
在IaaS层上解决这个问题一般来说有三种方案一种是使用比较廉价的开源产品NFS、Ceph、TiDB、CockroachDB、ElasticSearch、InfluxDB、MySQL Cluster和Redis Cluster之类的另一种是用云计算厂商的方案。当然如果不差钱的话可以使用更为昂贵的商业网络存储方案。
# 小结
回顾一下今天分享的主要内容。首先,我先明确表态,不要将流量调度和服务治理混为一谈(当然,服务治理是流量调度的前提),并比较了两者有何不同。
然后讲述了流量调度的主要功能和关键技术。接着进入本文的第二个话题——状态数据调度讲述了真正完整解决数据Scale问题的应该还是数据结点自身并给出了相应的技术方案随后对状态数据调度进行了小结。
欢迎你也谈一谈自己经历过的技术场景中是采用了哪些流量和数据调度的技术和产品,遇到过什么样的问题,是怎样解决的?
下篇文章中我们将开启一个全新的话题——洞悉PaaS平台的本质。
下面我列出了系列文章《分布式系统架构的本质》的目录,以方便你快速找到自己感兴趣的内容。如果你在分布式系统架构方面,有其他想了解的话题和内容,欢迎留言给我。
- [分布式系统架构的冰与火](https://time.geekbang.org/column/article/1411)
- [从亚马逊的实践,谈分布式系统的难点](https://time.geekbang.org/column/article/1505)
- [分布式系统的技术栈](https://time.geekbang.org/column/article/1512)
- [分布式系统关键技术:全栈监控](https://time.geekbang.org/column/article/1513)
- [分布式系统关键技术:服务调度](https://time.geekbang.org/column/article/1604)
- [分布式系统关键技术:流量与数据调度](https://time.geekbang.org/column/article/1609)
- [洞悉PaaS平台的本质](https://time.geekbang.org/column/article/1610)
- [推荐阅读:分布式系统架构经典资料](https://time.geekbang.org/column/article/2080)
- [推荐阅读:分布式数据调度相关论文](https://time.geekbang.org/column/article/2421)