之前有不少刚入坑 Java
的粉丝留言,想系统的学习一下分库分表相关技术,可我一直没下定决心搞,眼下赶上公司项目在使用 sharding-jdbc
对现有 MySQL
架构做分库分表的改造,所以借此机会出一系分库分表落地实践的文章,也算是自己对架构学习的一个总结。
我在网上陆陆续续的也看了一些有关于分库分表的文章,可发现网上同质化的资料有点多,而且知识点又都比较零碎,还没有详细的实战案例。为了更深入的学习下,我在某些平台买了点付费课程,看了几节课发现有点经验的人看还可以,但对于新手入门来说,其实学习难度还是蛮大的。
为了让新手也能看得懂,有些知识点我可能会用更多的篇幅加以描述,希望大家不要嫌我啰嗦,等这分库分表系列文章完结后,我会把它做成 PDF
文档开源出去,能帮一个算一个吧!如果发现文中有哪些错误或不严谨之处,欢迎大家交流指正。
具体实践分库分表之前在啰嗦几句,回头复习下分库分表的基础概念。
其实 分库
和 分表
是两个概念,只不过通常分库与分表的操作会同时进行,以至于我们习惯性的将它们合在一起叫做分库分表。
分库分表是为了解决由于库、表数据量过大,而导致数据库性能持续下降的问题。按照一定的规则,将原本数据量大的数据库拆分成多个单独的数据库,将原本数据量大的表拆分成若干个数据表,使得单一的库、表性能达到最优的效果(响应速度快),以此提升整体数据库性能。
分库分表的核心理念就是对数据进行切分(Sharding
),以及切分后如何对数据的快速定位与查询结果整合。而分库与分表都可以从:垂直
(纵向)和 水平
(横向)两种纬度进行切分。
下边我们就以订单相关的业务举例,看看如何做库、表的 垂直
和 水平
切分。
垂直切分有 垂直
分库 和 垂直
分表。
垂直分库相对来说是比较好理解的,核心理念就四个字:专库专用
。
按业务类型对表进行分类,像订单、支付、优惠券、积分等相应的表放在对应的数据库中。开发者不可以跨库直连别的业务数据库,想要其他业务数据,对应业务方可以提供 API
接口,这就是微服务的初始形态。
垂直分库很大程度上取决于业务的划分,但有时候业务间的划分并不是那么清晰,比如:订单数据的拆分要考虑到与其他业务间的关联关系,并不是说直接把订单相关的表放在一个库里这么简单。
在一定程度上,垂直分库似乎提升了一些数据库性能,可实际上并没有解决由于单表数据量过大导致的性能问题,所以就需要配合水平切分方式来解决。
垂直分表
是基于数据表的列(字段)为依据切分的,是一种大表拆小表的模式。
例如:一张 order
订单表,将订单金额、订单编号等访问频繁的字段,单独拆成一张表,把 blob
类型这样的大字段或访问不频繁的字段,拆分出来创建一个单独的扩展表 work_extend
,这样每张表只存储原表的一部分字段,再将拆分出来的表分散到不同的库中。
我们知道数据库是以行为单位将数据加载到内存中,这样拆分以后核心表大多是访问频率较高的字段,而且字段长度也都较短,因而可以加载更多数据到内存中,来增加查询的命中率,减少磁盘IO,以此来提升数据库性能。
垂直切分的优点:
垂直切分的缺点:
前边说了垂直切分还是会存在单库、表数据量过大的问题,当我们的应用已经无法在细粒度的垂直切分时,
依旧存在单库读写、存储性能瓶颈,这时就要配合水平切分一起了,水平切分能大幅提升数据库性能。
水平分库是把同一个表按一定规则拆分到不同的数据库中,每个库可以位于不同的服务器上,以此实现水平扩展,是一种常见的提升数据库性能的方式。
这种方案往往能解决单库存储量及性能瓶颈问题,但由于同一个表被分配在不同的数据库中,数据的访问需要额外的路由工作,因此系统的复杂度也被提升了。
例如下图,订单DB_1
、订单DB_1
、订单DB_3
三个数据库内有完全相同的表 order
,我们在访问某一笔订单时可以通过对订单的订单编号取模的方式 订单编号 mod 3 (数据库实例数)
,指定该订单应该在哪个数据库中操作。
水平分表是在同一个数据库内,把一张大数据量的表按一定规则,切分成多个结构完全相同表,而每个表只存原表的一部分数据。
例如:一张 order
订单表有 900万数据,经过水平拆分出来三个表,order_1
、order_2
、order_3
,每张表存有数据 300万,以此类推。
水平分表尽管拆分了表,但子表都还是在同一个数据库实例中,只是解决了单一表数据量过大的问题,并没有将拆分后的表分散到不同的机器上,还在竞争同一个物理机的CPU、内存、网络IO等。要想进一步提升性能,就需要将拆分后的表分散到不同的数据库中,达到分布式的效果。
水平切分的优点:
水平切分的缺点:
我们上边提到过很多次 一定规则
,这个规则其实是一种路由算法,就是决定一条数据具体应该存在哪个数据库的哪张表里。
常见的有 取模算法
和 范围限定算法
按字段取模(对hash结果取余数 (hash() mod N),N为数据库实例数或子表数量)是最为常见的一种切分方式。
还拿 order
订单表举例,先对数据库从 0 到 N-1进行编号,对 order
订单表中 work_no
订单编号字段进行取模,得到余数 i
,i=0
存第一个库,i=1
存第二个库,i=2
存第三个库....以此类推。
这样同一笔订单的数据都会存在同一个库、表里,查询时用相同的规则,用 work_no
订单编号作为查询条件,就能快速的定位到数据。
优点:
缺点:
按照 时间区间
或 ID区间
来切分,比如:我们切分的是用户表,可以定义每个库的 User
表里只存10000条数据,第一个库只存 userId
从1 ~ 9999的数据,第二个库存 userId
为10000 ~ 20000,第三个库存 userId
为 20001~ 30000......以此类推,按时间范围也是同理。
优点:
缺点:
由于表分布在不同库中,不可避免会带来跨库事务问题。一般可使用 "三阶段提交
"和 "两阶段提交
" 处理,但是这种方式性能较差,代码开发量也比较大。通常做法是做到最终一致性的方案,如果不苛求系统的实时一致性,只要在允许的时间段内达到最终一致性即可,采用事务补偿的方式。
这里我应用阿里的分布式事务框架Seata
来做分布式事务的管理,后边会结合实际案例。
分页、排序、联合查询是开发中使用频率非常高的功能,但在分库分表后,这些看似普通的操作却是让人非常头疼的问题。将分散在不同库中表的数据查询出来,再将所有结果进行汇总整理后提供给用户。
分库分表后数据库的自增主键意义就不大了,因为我们不能依靠单个数据库实例上的自增主键来实现不同数据库之间的全局唯一主键,此时一个能够生成全局唯一ID的系统是非常必要的,那么这个全局唯一ID就叫 分布式ID
。
不难发现大部分主流的关系型数据库都提供了主从架构的高可用方案,而我们需要实现 读写分离
+ 分库分表
,读库与写库都要做分库分表处理,后边会有具体实战案例。
数据脱敏,是指对某些敏感信息通过脱敏规则进行数据转换,从而实现敏感隐私数据的可靠保护,如身份证号、手机号、卡号、账号密码等个人信息,一般这些都需要进行做脱敏处理。
我还是那句话,尽量不要自己造轮子,因为自己造的轮子可能不那么圆,业界已经有了很多比较成熟的分库分表中间件,我们根据自身的业务需求挑选,将更多的精力放在业务实现上。
本文系作者在时代Java发表,未经许可,不得转载。
如有侵权,请联系nowjava@qq.com删除。