文章较长,听我娓娓道来。粉丝与关注,社交好友,都是典型的“多对多关系”的业务,这类业务的核心服务是好友中心,当关系链达到百亿之后,好友中心架构设计要考虑哪些因素,是本文将要分享的内容。什么是“多对多”关系?所谓的“多对多”,来自数据库设计中的“实体-关系”ER模型,用来描述实体之间的关联关系,一个学生可以选修多个课程,一个课程可以被多个学生选修,这里学生与课程时间的关系,就是多对多关系。什么是好友关系?好友关系主要分为两类:(1)弱好友关系;(2)强好友关系;两类都有典型的互联网产品应用。什么是弱好友关系?弱好友关系的建立,不需要双方彼此同意:用户A关注用户B,不需要用户B同意,此时用户A与用户B为弱好友关系,对A而言,他多“关注”了一个人,对B而言,他多了一个“粉丝”。微博粉丝是一个典型的弱好友关系应用。什么是强好友关系?强好友关系的建立,需要好友关系双方彼此同意:用户A请求添加用户B为好友,用户B同意,此时用户A与用户B则互为强好友关系,即A是B的好友,B也是A的好友。QQ好友是一个典型的强好友关系应用。什么是好友中心?好友中心是一个典型的多对多业务,一个用户可以添加多个好友,也可以被多个好友添加。其典型架构如上:(1)friend-service:好友中心服务,对调用者提供友好的RPC接口;(2)db:对好友数据进行存储;服务的接口,不外乎:关注,取关,增加好友,删除好友,同意好友申请,不同意好友申请。其核心,在于元数据的设计。弱好友关系,如何设计元数据?通过弱好友关系业务分析,很容易了解到,其核心元数据为:(1)guanzhu(uid, guanzhu_uid);(2)fensi(uid, fensi_uid);其中:(1)guanzhu表,用户记录uid所有关注用户guanzhu_uid;(2)fensi表,用来记录uid所有粉丝用户fensi_uid;需要强调的是,一条弱关系的产生,会产生两条记录,一条关注记录,一条粉丝记录。画外音:可不可以只有一条记录?例如:用户A(uid=1)关注了用户B(uid=2),A多关注了一个用户,B多了一个粉丝,于是:(1)guanzhu表要插入{1, 2}这一条记录,1关注了2;(2)fensi表要插入{2, 1}这一条记录,2粉了1;如何查询一个用户关注了谁呢?在guanzhu的uid上建立索引:select * from guanzhu where uid=1;即可得到结果,1关注了2。如何查询一个用户粉了谁呢?在fensi的uid上建立索引:select * from fensi where uid=2;即可得到结果,2粉了1。强好友关系,如何设计元数据?通过强好友关系业务分析,很容易了解到,其核心元数据为:(1)friend(uid1, uid2);其中:(1)uid1,强好友关系中一方的uid;(2)uid2,强好友关系中另一方的uid;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=2unionselect * from friend where uid2=2即可得到结果。画外音,可不可以使用: select * from friend uid1=2 or uid2=2使用一个表记录所有关系链,如果数据量大了,数据库进行分库以后,不久无法同时满足uid1和uid2上的查询了么,此时要怎么办呢?此时,可以使用类似于弱关系实现的方案,用数据冗余的方式,即使分库后,依然能够满足两种查询需求。即,强好友关系也可以使用关注表和粉丝表来实现:(1)guanzhu(uid, guanzhu_uid);(2)fensi(uid, fensi_uid);例如:用户A(uid=1)和用户B(uid=2)为强好友关系,即相互关注:用户A(uid=1)关注了用户B(uid=2),A多关注了一个用户,B多了一个粉丝,于是:(1)guanzhu表要插入{1, 2}这一条记录;(2)fensi表要插入{2, 1}这一条记录;同时,用户B(uid=2)也关注了用户A(uid=1),B多关注了一个用户,A多了一个粉丝,于是:(1)guanzhu表要插入{2, 1}这一条记录;(2)fensi表要插入{1, 2}这一条记录;强调一下:数据冗余,是多对多关系,满足不同维度的查询需求,在数据量大时,数据水平切分的常用实践。对于强好友关系的两类实现:第一类:friend(uid1, uid2)表;第二类:数据冗余guanzhu表与fensi表(后文称正表T1与反表T2);在数据量小时,看似无差异,但数据量大时,只有后者,才能满足两类查询需求:(1)friend表,数据量大时,如果使用uid1来分库,那么uid2上的查询就需要遍历多库;(2)正表T1与反表T2通过数据冗余来实现好友关系,{1, 2}{2,1}分别存在于两表中,故两个表都使用uid来分库,均只需要进行一次查询,就能找到对应的关注与粉丝,而不需要多个库扫描;问题转化为,T1和T2正反表,如何进行数据冗余呢?数据冗余,常见有三种方法。方法一:服务同步冗余