Anna(支持任意扩展和超高性能的KV数据库系统)阅读笔记

前言

年前被同事安利了这个分布式最终一致性的存储系统 Annaarrow-up-right 。初略看了一眼Paper,似乎很是牛X。说是支持任意规模的扩展,并且性能不低于 pedisarrow-up-right。于是抽空来看看并了解下这套系统的设计特点和这种夸张的单机性能和扩展性的来源。

主流分布式KVS的比较

系统名词

扩容设计

内存模型

针对单个Key的一致性策略

针对多个Key一致性策略

Masstree

多核

共享内存

Bw-tree

多核

共享内存

PALM

多核

共享内存

MICA

多核

共享内存

PNUTS

分布式

消息队列

线性写, 单调读

CouchDB

分布式

消息队列

HBase

分布式

消息队列

Riak

分布式

消息队列

Memcached

多核&分布式

共享内存&消息队列

MongoDB

多核&分布式

共享内存&消息队列

一致性说明:

See https://en.wikipedia.org/wiki/Consistency_modelarrow-up-right for more details

Annaarrow-up-right 架构设计

Annaarrow-up-right 整体上依赖 bounded join semilatticearrow-up-right (文中简称 latticearrow-up-right ) 的核心设计,不知道该怎么翻译了。

数学上 Semilatticearrow-up-right 符合 ACI 特性:

  • 交换律(Commutativity): \( \sqcup (a, b) = \sqcup (b, a), \forall_{a,b} \in S \) ,即操作和顺序无关

  • 结合律(Associativity): \( \sqcup (\sqcup (a, b), c) = \sqcup (a, \sqcup (b, c)), \forall_{a,b,c} \in S \) ,即操作和先后次序无关

  • 幂等性(Idempotence): \( \sqcup (a, a) = a, \forall_{a} \in S \) ,即操作多次不影响结果

  • join-semilattice : join操作 \( \sqcup (a, b) \) 指a,b的最小上界。

    这里是数学定义,我的理解在 Annaarrow-up-right 中指数据合并操作。

  • bounded : 指存在特定元素 1,使得 \( \sqcup (a, 1) = a , \forall_{a,1} \in S \)

详见维基百科: https://en.wikipedia.org/wiki/Semilatticearrow-up-right

我的理解就是一个比较细粒度的模块化数据块,互相之间没有强交互和关联,大多数情况下仅处理自己的数据集。然后对于数据合并的操作,要设计成符合上面的 ACI 特性。

latticearrow-up-right 的基础上,很容易就可以设计出适合这种场景的分布式状态模型和通信模型。

整个架构使用了actor模型。和 seastararrow-up-right 框架相似,系统采用多线程结构,并且按CPU核心数分配线程。并且和 golang 的核心思想一样,用 Message-passingshared-memory 。 这样虽然是多线程,但是每个线程跑自己的 Actor Core ,在业务处理的时候互相之间几乎不需要互相通信。每个actor线程由自己的数据变更集(changeset),然后定期访问广播通道执行集群管理或者是合并操作。

1905-01.png

Annaarrow-up-right 的key和节点的分布也是一致性哈希。采用 gossiparrow-up-right 算法来处理actor的容灾和扩缩容。按文中的意思似乎数据分片的同步也采用的是 gossiparrow-up-right 。不过在这一点上 gossiparrow-up-right 对分布式系统的局部分片扩容天生是比 Paxosarrow-up-rightRaftarrow-up-right 要好很多。

另外在 Annaarrow-up-right 的actor中,记录了所有其他actor最后感知到这个actor事件的 向量时钟(Vector Clock)arrow-up-right ,这样在多副本时,比如出现任意副本对某个Key的删除操作,就可以用因果关系感知到其他副本的actor都同步了这个事件之后再执行真正的垃圾回收操作。这就避免了不一致的时候再被同步回来的问题。这个事件最差情况也会由广播机制定期同步。

其实我有点怀疑这种方案在大规模集群上的延迟,这样意味着每个actor两两之间都需要比较高密度的数据同步。后面给出的各项测试模式里也没有涉及大量节点和大规模跨机器的测试结果。大多数给的都是少数机器几十个节点的性能报告。

弹性可扩展的一致性策略

Annaarrow-up-right 是通过设计成一个无需交互,并且让内部符合ACI特性来实现这个高性能高并发的KVS的。Annaarrow-up-right 通过一种自底向上的方法,通过把复杂数据结构切割成一个一个的 ACI Building Blocks(Lattice)arrow-up-right 来保证整体的保持ACI特性。系统内置了一些一致性模型的设计,然后用户也可以自定义Merge函数。 Annaarrow-up-right 的设计重点之一就是方便用户可以用 Annaarrow-up-right 已有提供好的符合 ACIlatticearrow-up-right 来组合成自己需要的也符合 ACI 的新形态。然后以此来很方便地实现上面提到的那么多种一致性策略。

文中说是以C++模板来实现易扩展的 latticearrow-up-right 的。感觉和STL的思路比较像。

比如说实现了 MapLattice (KV结构) 、 PairLattice (带版本号的数据对) 、 MaxIntLattice (递增整数) 、 ValueLattice (用户自定义数据)。可以按下面的结构实现 因果一致性(Causal)arrow-up-right

1905-02.png

如图,使用Key为 ClientID, Value为 MaxIntLatticeMapLattice 来实现因果一致性所需的 向量时钟(Vector Clock)arrow-up-right 。 大部分情况下,分片稳定并且系统运行良好的情况下,数据的因果关系可以通过 向量时钟(Vector Clock)arrow-up-right 判定出来。少数情况下,如果发生了冲突,就要走冲突合并流程了。只要合并操作符合上面的 ACI 特性,等到一定时间窗以后,无论哪个节点先执行合并、数据合并的顺序是怎样,一定会有一个统一的最终结果。

比如合并数据的时候采用按时间+节点ID排序,然后舍弃老数据。就是无论怎么合并。最终的结果一定是一致的。

再比如 Read Committedarrow-up-right 的一致性策略。 其他的部分和 因果一致性(Causal)arrow-up-right 一样,只要把维护Key的 向量时钟(Vector Clock)arrow-up-right 改成用 MaxIntLattice 表示的最大时间戳,然后合并策略改成按最新时间戳就行了。

性能

Annaarrow-up-right 对比其他的 KVS 都是碾压的,但是我个人对其他 KVS的设计不熟,对 redisarrow-up-right 倒是比较熟悉。所以这里贴出来它和 redisarrow-up-right (主要是 redis-cluster) 的对比。

1905-03.png

如图,在低争用的情况下,Annaarrow-up-right 的表现和 redis-clusterarrow-up-right 相近。在高争用的时候,这里说是 Annaarrow-up-right 的优势比较明显。但是我个人的理解, Annaarrow-up-right 的这部分性能完全来自于对一致性的舍弃上。 redis-clusterarrow-up-right 是很容易配置成强一致,但是 Annaarrow-up-right 是采用了多个 Actor Core 独立运行,然后最终进行Merge的形式。那么在高争用的时候, redis-clusterarrow-up-right 相当于收敛到了少数节点上运行,但是仍然是可以实现保证一致性的;而 Annaarrow-up-right 是先跑完再Merge,这种的话一致性就比较差,也很难实现CAS类的操作(否则也会fallback到少数节点执行,那么和 Annaarrow-up-right 的设计模式就相背离了)。

它在性能消耗分析的部分也说明了,如果一致性策略选的比较弱的话,多副本高争用的情况下还会有Merge开销激增的情况。这个Merge因为是跨节点通信的操作,在Paper提供的3副本的部署结构下大约69%的CPU消耗在这上面。所以影响还是蛮大的,相当于单点性能退化了。

总结

最终一致性和强一致性的数据库系统还是有很大区别,最终一致性不能保证大家看到的中间状态是一致的,并且可能需要自己去选择或者提供冲突时的处理方法。感觉上可能比较适合非关键性数据的分发和存储。至少在游戏项目中,感觉更实用的还是能保证强一致性的系统,即便是NoSQL系统。因为很多东西有比较复杂的逻辑关系,不太能接受多个节点结果不一致的状态。

上面 Annaarrow-up-right 和其他系统的比较里也有列出一些强一致性的系统,我觉得和这些强一致的系统比性能其实不太公平。但是 Annaarrow-up-right 的一些设计层面的东西还是值得参考的,比如它对worker的设计和考量。这方面它和 pedisarrow-up-right 也很像,而且 pedisarrow-up-right 性能比它没差太多,但是我的理解没错的话它可以保持强一致,感觉更胜一筹。当然 pedisarrow-up-right 功能上目前实现也还是弱了一些,事务的支持和CAS的支持都没有。也就是说乐观锁和悲观锁都还不支持,目前还是没法用于要求强一致的业务里,也就和 Annaarrow-up-right 没太大区别了。

从分布式设计来看,Annaarrow-up-rightredis-clusterarrow-up-right 的大体结构是一样的。节点间通信都是消息传递,节点维护都是 gossiparrow-up-right ,节点收到其他节点的请求都是redirect过去。区别是 Annaarrow-up-right 使用的多线程设计, 而 redis-clusterarrow-up-right 是多进程模型;Annaarrow-up-right 提供了更完备的proxy层,而redis-cluster目前这方面还得靠客户端支持;然后 Annaarrow-up-right 对于resharding的自动化更好一些,并且还实现了热Key的动态副本。如果单单从单点性能的话,Annaarrow-up-right 是和redis差不多的。

个人感觉 Annaarrow-up-right 最大的优势还是在于它的 latticearrow-up-right 的设计。这极大地方便了对多种一致性策略的扩展。以后如果有什么新的技术方案出来也能够比较容易在这上面实现和测试。

[22]: https://ai.google/research/pubs/pub39966arrow-up-right "Spanner: Google's Globally-Distributed Database"

Last updated

Was this helpful?