来自 GitChat 做者:沈剑
更多IT技术分享,尽在微信公众号:GitChat技术杂谈git
本文将以“好友中心”为例,介绍“多对多”类业务,随着数据量的逐步增大,数据库性能显著下降,数据库水平切分相关的架构实践。数据库
所谓的“多对多”,来自数据库设计中的“实体-关系”ER模型,用来描述实体之间的关联关系,一个学生能够选修多个课程,一个课程能够被多个学生选修,这里学生与课程时间的关系,就是多对多关系。微信
好友关系主要分为两类,弱好友关系与强好友关系,两类都有典型的互联网产品应用。架构
弱好友关系的创建,不须要双方彼此赞成:异步
微博粉丝是一个典型的弱好友关系应用。数据库设计
强好友关系的创建,须要好友关系双方彼此赞成:分布式
QQ好友是一个典型的强好友关系应用。工具
好友中心是一个典型的多对多业务,一个用户能够添加多个好友,也能够被多个好友添加,其典型架构为:性能
经过弱好友关系业务分析,很容易了解到,其核心元数据为:学习
guanzhu(uid, guanzhu_uid);
fensi(uid, fensi_uid);
其中:
须要强调的是,一条弱关系的产生,会产生两条记录,一条关注记录,一条粉丝记录。
例如:用户A(uid=1)关注了用户B(uid=2),A多关注了一个用户,B多了一个粉丝,因而:
如何查询一个用户关注了谁呢?
回答:在guanzhu的uid上创建索引:
select * from guanzhu where uid=1;
便可获得结果,1关注了2。
如何查询一个用户粉了谁呢?
回答:在fensi的uid上创建索引:
select * from fensi where uid=2;
便可获得结果,2粉了1。
经过强好友关系业务分析,很容易了解到,其核心元数据为:
friend(uid1, uid2);
其中:
uid=1的用户添加了uid=2的用户,双方都赞成加彼此为好友,这个强好友关系,在数据库中应该插入记录{1, 2}仍是记录{2,1 }呢?
回答:均可以,为了不歧义,能够人为约定,插入记录时uid1的值必须小于uid2。
例如:有uid=1,2,3三个用户,他们互为强好友关系,那边数据库中多是这样的三条记录
{1, 2}
{2, 3}
{1,3 }
如何查询一个用户的好友呢?
回答:假设要查询uid=2的全部好友,只需在uid1和uid2上创建索引,而后:
select * from friend where uid1=2
union
select * from friend where uid2=2
便可获得结果。
做业:为什么不使用这样的SQL语句呢?
select * from friend uid1=2 or uid2=2
供你们思考。
强好友关系是弱好友关系的一个特例,A和B必须互为关注关系(也能够说,同时互为粉丝关系),即也可使用关注表和粉丝表来实现:
guanzhu(uid, guanzhu_uid);
fensi(uid, fensi_uid);
例如:用户A(uid=1)和用户B(uid=2)为强好友关系,即相互关注:
用户A(uid=1)关注了用户B(uid=2),A多关注了一个用户,B多了一个粉丝,因而:
同时,用户B(uid=2)也关注了用户A(uid=1),B多关注了一个用户,A多了一个粉丝,因而:
对于强好友关系的两类实现:
在数据量小时,看似无差别,但数据量大时,数据冗余的优点就体现出来了:
数据冗余,是多对多关系,在数据量大时,数据水平切分的经常使用实践。
接下来的问题转化为,好友中心服务如何来进行数据冗余,常见有三种方法。
顾名思义,由好友中心服务同步写冗余数据,如上图1-4流程:
优势:
缺点:
若是系统对处理时间比较敏感,引出经常使用的第二种方案
数据的双写并再也不由好友中心服务来完成,服务层异步发出一个消息,经过消息总线发送给一个专门的数据复制服务来写入冗余数据,如上图1-6流程:
优势:
请求处理时间短(只插入1次)
缺点:
若是想解除“数据冗余”对系统的耦合,引出经常使用的第三种方案
数据的双写再也不由好友中心服务来完成,而是由线下的一个服务或者任务来完成,如上图1-6流程:
优势:
缺点:
上述三种方案各有优缺点,能够结合实际状况选取。
数据冗余当然可以解决多对多关系的数据库水平切分问题,但又带来了新的问题,如何保证正表T1与反表T2的数据一致性呢?
上一节的讨论能够看到,无论哪一种方案,由于两步操做不能保证原子性,总有出现数据不一致的可能,高吞吐分布式事务是业内还没有解决的难题,此时的架构优化方向,并非彻底保证数据的一致,而是尽早的发现不一致,并修复不一致。
最终一致性,是高吞吐互联网业务一致性的经常使用实践。更具体的,保证数据最终一致性的方案有三种。
如上图所示,线下启动一个离线的扫描工具,不停的比对正表T1和反表T2,若是发现数据不一致,就进行补偿修复。
优势:
缺点:
有没有只扫描“可能存在不一致可能性”的数据,而不是每次扫描所有数据,以提升效率的优化方法呢?
每次只扫描增量的日志数据,就可以极大提升效率,缩短数据不一致的时间窗口,如上图1-4流程所示:
固然,咱们仍是须要一个离线的扫描工具,不停的比对日志log1和日志log2,若是发现数据不一致,就进行补偿修复
优势:
缺点:
有没有实时检测一致性并进行修复的方法呢?
此次不是写日志了,而是向消息总线发送消息,如上图1-4流程所示:
此次不是须要一个周期扫描的离线工具了,而是一个实时订阅消息的服务不停的收消息。
假设正常状况下,msg1和msg2的接收时间应该在3s之内,若是检测服务在收到msg1后没有收到msg2,就尝试检测数据的一致性,不一致时进行补偿修复
优势:
缺点:
however,技术方案自己就是一个投入产出比的折衷,能够根据业务对一致性的需求程度决定使用哪种方法。
文字较多,但愿尽可能记住以下几点:
冗余数据的常见方案有三种:
最终一致性的常见实践是,尽快找到不一致,并修复数据,常见方案有三种。
以订单中心为典型的“多KEY”类业务的水平拆分架构又应该怎么处理,敬请期待下期。
咱们 GitChat 通过半年的运营,在产品形态上不断改进。
此次,咱们将系统性的优化 GitChat 一些内容,在原有 Chat 基础上推出新的内容分享产品:
GitQ 精品课程
若是您在 IT技术上有本身独到的学习心得 或 自成体系的技术成长套路,咱们很是期待您能在 GitQ 上进一步实现:
GitQ 定位:具有专业性、成体系的IT类课程
GitQ 形态:独家文章(6~12篇)+读者圈答疑
GitQ 订价:9.99 / 19.99 等
【做者如何开设 GitQ 精品课?】
【做者注意事项】
【更多咨询】
在微信公众号: GitChat技术杂谈(GitChat_Club) 留言 ,或加微信联系 GitChat 主编:ztx415