You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

173 lines
16 KiB
Markdown

2 years ago
# 10 | 分布式体系结构之非集中式结构:众生平等
你好,我是聂鹏程。今天,我来继续带你打卡分布式核心技术。
在上一篇文章中,我带你了解了分布式体系结构中的集中式结构。虽然很多云上的管理都采用了集中式结构,但是这种结构对中心服务器性能要求很高,而且存在单点瓶颈和单点故障问题。
为了解决这个问题,分布式领域中又出现了另一个经典的系统结构,即非集中式结构,也叫作分布式结构。那什么是非集中式结构呢,它的原理是什么样的,又有哪些集群采用了这种结构呢?
今天,我们就一起打卡非集中式结构,揭开它的神秘面纱吧。
## 什么是非集中式结构?
在非集中式结构中,服务的执行和数据的存储被分散到不同的服务器集群,服务器集群间通过消息传递进行通信和协调。
![](https://static001.geekbang.org/resource/image/f6/5e/f6c9e7033f09f240e21877b11170675e.png)
也就是说,在非集中式结构中,没有中央服务器和节点服务器之分,所有的服务器地位都是平等(对等)的,也就是我们常说的“众生平等”。这样一来,相比于集中式结构,非集中式结构就降低了某一个或者某一簇计算机集群的压力,在解决了单点瓶颈和单点故障问题的同时,还提升了系统的并发度,比较适合大规模集群的管理。
所以近几年来Google、 Amazon、Facebook、阿里巴巴、腾讯等互联网公司在一些业务中也相继采用了非集中式结构。
接下来我将为你介绍3种典型的非集中式架构系统包括Akka集群、Redis集群和Cassandra集群来帮助你深入理解非集中式架构。
## Akka集群
在介绍Akka集群的结构之前我带你了解一下什么是Akka框架吧。
Akka是一个开发库和运行环境用于构建可扩展的、弹性的、快速响应的应用程序。Akka框架是基于Actor模型实现的Actor模型是一个封装了状态和行为的对象它接收消息并基于该消息执行计算。Actor之间通信的唯一机制就是消息传递每个Actor都有自己的MailBox。
比如在分布式系统中一个服务器或一个节点可以视为一个ActorActor与Actor之间采用mail进行通信如下图所示
![](https://static001.geekbang.org/resource/image/3d/11/3d64427315a82ef08f4dd0fc88d9a311.png)
可以看到Actor发送的Mail消息会存储在接收方的MailBox中。默认情况下接收方按照mail到达的先后顺序从MailBox中提取mail消息并进行相应的计算处理。
> 备注关于Actor模型更详细的内容我会在第17篇文章中与你讲述。
显然Actor模型采用异步消息调用机制具有非阻塞、高性能等特点可以用于处理并发问题。Akka集群充分利用了Actor模型的优势提供了一个非集中式架构的集群管理模块用来构建可扩展的、弹性的分布式应用程序。
Akka集群负责Actor模型底层的节点管理包括故障检测、节点加入/退出集群等。也就是说Akka集群为Actor模型提供了一个可容错、去中心化的节点集群管理系统来保证Actor的运行和Actor之间的通信。
如下图所示Akka集群是一个完全去中心化的分布式集群管理系统。一个集群由多个节点组成每个节点都可以进行数据处理和任务执行节点之间均可进行通信。节点有Leader节点和非Leader节点之分。与非Leader节点相比**Leader节点只是增加了负责节点的加入和移除集群的功能**,所以并不会影响非集中式结构中节点的平等关系。
![](https://static001.geekbang.org/resource/image/83/bf/83cbb5e70a1d7dc628f4b0cf1ec658bf.png)
可以看到Akka集群的两个重点是数据传输和集群组建及管理所以接下来我将从这两个方面与你介绍Akka集群。
**首先,我们看一下数据传输。**在Akka集群中节点是对等的也就是说每个节点是可以并发处理的因此必然存在数据传输和一致性的问题。
比如我们要针对数据进行操作将X=1修改为X=2。现在集群中节点1进行了修改使得X=2但其他节点上还是X=1因此节点1需要将X=2的消息告知其他节点以保证最终集群中所有节点上均为X=2。
其实这个问题就是分布式共识问题。我已经在第5篇文章“[分布式共识:存异求同](https://time.geekbang.org/column/article/144548)”中与你介绍了PoW、PoS和DPoS三种达成共识的方法你可以再复习下相关内容。
**Akka集群主要采用的是谁的时间戳最新也就是数据最新就以谁为准的原则。**在这里我要重点与你讲述的是如何将X=2这个消息传输给集群中的每个节点。
Akka集群采用了**Gossip协议**该协议是最终一致性协议。它的原理是每个节点周期性地从自己维护的集群节点列表中随机选择k个节点将自己存储的数据信息发给这k个节点接收到该信息的节点采用前面讲的共识原则对收到的数据和本地数据进行合并这样迭代几个周期后集群中所有节点上的数据信息就一致了。
这就好比我们生活中的“谣言传播”一样用户A告诉用户B“商场新开了一家火锅店”用户B收到信息后再告诉用户C然后用户C再告诉用户D。这样用户A、B、C、D最终都知道了这个消息。
**接下来,我们看一下集群组建及管理。**下图展示了Akka集群的创建过程。在创建集群时节点被分为三种类型
* 种子节点。使用静态配置文件方式或者系统运行时指定方式,可以生成种子节点;种子节点是普通节点加入集群的联系点,可以自动接收新加入集群的节点的信息。
* 首种子节点。首种子节点是配置文件中的第一个种子节点其功能是集群第一次启动时首种子节点启动起来集群才能组建成功保证集群第一次创建时只有一个集群。如下图A节点就是Akka集群的首种子节点。
* 普通节点。可以向种子节点或集群中的任意节点发送Join消息请求加入集群。如下图的B和C节点通过向A节点发送Join消息从而加入到Akka集群。
![](https://static001.geekbang.org/resource/image/fe/c8/fed5d41f891df279f114ca4492c5e1c8.png)
> 图片来源:[https://getakka.net/articles/clustering/cluster-overview.html](https://getakka.net/articles/clustering/cluster-overview.html)
Akka集群的每个节点启动后读取配置文件获取种子节点列表然后开始组建集群
* 如果本节点为首种子节点,则把自己加入到集群列表中,即以自己为中心构建集群;
* 如果本节点为种子节点,则向首种子节点请求加入集群,当首种子节点回复同意消息后,可以加入集群,否则不可加入集群;
* 如果本节点为普通节点,则可以向任一种子节点(包括首种子节点)请求加入集群,收到同意后,则加入集群,否则不可加入集群。
加入首种子节点或种子节点的节点信息会通过Gossip协议的传播方式传播给当前已加入的所有节点以完成集群组建。当集群组建完成后就不存在种子节点与普通节点之分了每个节点均可执行Actor应用程序。
Akka集群可以构建可扩展的、弹性的分布式应用程序因此在JVM中应用了Akka框架从而实现并发编程。目前豌豆荚、蘑菇街等公司采用了Akka集群。
**到这里,我们小结一下吧**。Akka集群是一个完全去中心化的集群管理系统当集群组建完成后每个节点均可执行Actor应用程序因此支持并发操作。但这个并发操作引入了数据同步和一致性问题所以Akka集群采用了Gossip协议进行数据同步通过谁的时间戳最新就以谁为准来解决一致性问题。
在实际业务场景中除了面向应用程序平台的分布式集群管理之外分布式数据存储也是一个非常重要的话题。在这其中分布式数据存储中的集群管理便是一个关键因素。那么接下来我就以开源数据库Redis的集群管理系统为例与你展开介绍吧。
## Redis集群
Redis是一个开源的、包含多种数据结构的高性能Key-value数据库主要有以下特征
* 支持多种数据结构包括字符串String、散列Hash、列表List、集合Set、有序集合Sorted Set
* 支持数据的持久化和备份。数据可以保存在磁盘中下次直接加载使用且可以采用主从模式Master/Slave进行数据备份。
* 基于内存运行,具有极高的性能。
Redis的这些特征均是为数据存储进行服务的数据可分片存储在不同的Redis节点上多个Redis节点间可共享数据而提供这项能力的就是Redis 集群。
Redis 集群中不存在中央节点是典型的去中心化结构每个节点均可与其他节点通信。所有节点均可负责存储数据、记录集群的状态包括键值到正确节点的映射客户端可以访问或连接到任一节点上。Redis 集群的架构图,如下所示。
当然节点之间的数据传输仍采用了Gossip协议来保证集群中数据的最终一致性。
![](https://static001.geekbang.org/resource/image/dd/71/ddfc9c0ba59b44f975e720544f173371.png)
Redis集群中的节点用于数据存储所以**在设计时,需要考虑数据的可靠性和分片存储问题**。
对于可靠性的问题集群中每个节点均存在主备也就是说每台服务器上都运行两个Redis服务分别为主备主故障后备升主。
而对于数据的分片存储问题Redis集群引入了**“哈希槽”**的这一概念。Redis 集群内置了16384个哈希槽每个节点负责一部分哈希槽。当客户端要存储一个数据或对象时Redis先对key进行CRC16校验然后进行16384取模也即HASH\_SLOT = CRC16(key) mod 16384来决定哈希槽的编号从而确定存储在哪个节点上。
比如当前集群有3个节点那么:
* 节点 A 包含 0 到 5500号哈希槽
* 节点 B 包含5501 到 11000 号哈希槽;
* 节点 C 包含11001 到 16383号哈希槽。
Redis集群利用哈希槽实现了数据的分片存储从而将Redis的写操作分摊到了多个节点上提高了写并发能力。
**到这里,我们小结一下。**Redis集群是一个非集中式集群管理系统没有中心节点不会因为某个节点造成性能瓶颈每个节点均支持数据存储且采用分片存储方式提高了写的并发能力。同时每个节点的设计采用主备设计提高了数据的可靠性。
鉴于这些优点Redis已被Twitter、Uber、GitHub、Instagaram等公司采用。
除了Redis外还有一个开源分布式key-value数据库系统Cassandra。接下来我就再与你分享下Cassandra集群的设计以加深你对非集中式架构的理解。
## Cassandra集群
与Redis类似Cassandra也支持数据的分布式存储和操作。因此Cassandra的集群架构与数据分片存储方案与Redis集群类似。
如下图所示Cassandra集群的系统架构是基于一致性哈希的完全P2P结构没有Master的概念所有节点都是同样的角色彻底避免了因为单点问题导致的系统不稳定。Cassandra集群节点间的状态同步也是通过Gossip协议来进行P2P通信的。
![](https://static001.geekbang.org/resource/image/a6/c4/a6f5b593bb407d445213e2b2c102dfc4.png)
集群中的每个节点,都可以存储数据,并接收来自客户端的请求。**Cassandra集群数据存储与Redis的不同之处是**Redis集群每个节点代表一部分哈希槽一个哈希槽代表一个哈希值区间而Cassandra集群中每个节点代表一个哈希值。
在Cassandra集群中每次客户端可以向集群中的任意一个节点请求数据接收到请求的节点将key值进行哈希操作找出在一致性哈希环上是哪些节点应该存储这个数据然后将请求转发到相应节点上并将查询结果反馈返回给客户端。
目前Cassandra集群因为完全去中心化的结构模式已经被Hulu、Apple、Comcast、Instagram、Spotify、eBay、Netflix等公司使用。
**到这里,我们小结一下吧。**Cassandra采用去中心化的架构解决了集中式结构的单点故障问题同时因为数据基于哈希值分区存储提高了读写数据的并发能力。在Cassandra集群中没有Master的概念每个节点代表一个哈希值通过哈希映射的方式决定数据存储的位置。集群间的状态同步通过Gossip协议来进行P2P的通信。
## 对比分析
好了以上就是Akka集群、Redis集群和Cassandra集群的主要内容了。为了便于理解与记忆我将这3个集群的主要特征梳理为了一张表格如下所示
![](https://static001.geekbang.org/resource/image/be/88/be7d5a5b7e435f34fc60b5bf9a54a988.jpg)
## 知识扩展如何优化Gossip协议中的重复消息问题
非集中式结构的通信协议采用了Gossip协议。而Gossip是一种谣言传播协议每个节点周期性地从节点列表中选择k个节点将本节点存储的信息传播出去直到所有节点信息一致即算法收敛了。
这里有个问题如果每次都是随机选择k个节点的话势必会存在重复选择同样节点的可能增加消息量。你觉得这个问题是否可以优化又应该如何优化呢
首先这个问题肯定是可以优化的。解决方案是每个节点记录当前传输的消息且还未达到收敛的时候已经发送给了哪些节点然后每次选择时从没有发送过的节点列表中随机选择k个节点直到所有节点均被传输或集群收敛为止。这样一方面减少了重复消息量另一方面加快了收敛速度。
## 总结
集中式结构虽然易于理解但容易出现单点瓶颈和单点故障等问题而非集中结构才是超大规模分布式系统的首选结构。所以今天我以Akka集群、Redis集群和Cassandra集群的结构为例与你详细介绍了非集中式架构。
Akka集群是一个完全去中心化的集群管理系统节点之间都是P2P的连接模式通过Gossip协议来进行通信节点之间有角色划分负责数据存储的节点会进行存储数据。
Redis集群也是P2P的网状连接模式但是基于key-value的数据库模型每个节点都可以执行数据的计算和存储。此外Redis集群引入了哈希槽的概念来解决数据的分片存储问题。
Cassandra集群的结构是一致性哈希的P2P节点会构成一个环结构通过哈希映射来选择对应的节点。
好了,到最后,我再以一个思维导图为你总结一下这三个集群核心知识点,以方便你理解与记忆。
![](https://static001.geekbang.org/resource/image/00/67/00b66db06581dd66e5f33f8100d78f67.png)
虽然这三种集群的节点组织结构各有不同但节点之间都是通过Gossip协议来传递信息的。因此在实现过程中集群的消息传输、节点的功能等在不同的分布式系统中都是类似的而难点主要在于集群结构的设计。
由于Akka集群、Redis集群和Cassandra集群都是典型的非集中式集群组织结构目前应用已经非常广泛了所以有很多的实现案例可供你借鉴了。对于具体集群使用配置可参考相应的官网手册讲得比较全和细。
相信你通过对今天的学习,很快就可以上手非集中式架构的集群管理应用和实践了。加油,挑战一下自己吧!
## 思考题
边缘计算中边缘设备的管理,你认为适合非集中式结构还是集中式结构呢,原因又是什么呢?
我是聂鹏程,感谢你的收听,欢迎你在评论区给我留言分享你的观点,也欢迎你把这篇文章分享给更多的朋友一起阅读。我们下期再会!