概述
在大数据时代,数据量的增大使得单机存储已经无法满足需求,因此出现了分布式存储。MongoDB 也不例外,它大力推崇的分布式存储方式是分片集群。本文将介绍 MongoDB 分片集群的搭建方法及其实现原理。
分片集群是什么
MongoDB 分片集群是指将一个大的数据库分散到多个 shard(分片)上,实现数据的分布式存储,同时也解决了单机存储的瓶颈问题。在分片集群中,每一个 shard 都是一个完整的 MongoDB 实例,负责存储某一部分数据。而 Mongos(MongoDB 路由器)负责将所有数据的查询请求路由到对应的 shard 上。如下图:
整个集群由三个部分组成:
- Config Server
- Shard
- Mongos
Config Server 存储了整个集群的配置,每个 shard 负责存储一个片段数据,而 Mongos 指导应用程序的查询请求路由到合适的 shard 上。
搭建 MongoDB 分片集群
1. 确认环境
- 确认各个节点能够互相通信,建议使用内网,提高稳定性和性能;
- 确认每个节点都安装了 MongoDB;
- 文章中使用的 MongoDB 版本为 4.4.8。
2. 启动 Config Server
在 MongoDB 官方文档 (https://docs.mongodb.com/v4.4/tutorial/deploy-shard-cluster/) 中,提供了启动 Config Server 的脚本:
mongod --configsvr --replSet configReplSet --bind_ip localhost,hostname --port 27019 --dbpath /data/configdb1,/data/configdb2,/data/configdb3
其中,--replSet 指定了集群名称为 configReplSet,--bind_ip 指定了 bind 的 IP 地址,--port 指定了端口号,--dbpath 指定了存储实例数据的目录。需要注意的是,此处的目录需要是一个共享目录,因为 Config Server 是集群的基础架构,必须高可用。如下是启动 Config Server 的示例命令:
mongod --configsvr --replSet configReplSet --bind_ip 192.168.1.101,192.168.1.102,192.168.1.103 --port 27019 --dbpath /data/configdb1,/data/configdb2,/data/configdb3
3. 启动 Shard
在 MongoDB 中,每个 shard 是一个完整的 MongoDB 实例。因此,启动 shard 与单节点启动 MongoDB 时的过程是相同的。添加 --shardsvr 选项启动 shard。如下是启动 shard 的示例命令:
mkdir /data/mongodb/db mongod --shardsvr --replSet shardReplSet1 --bind_ip localhost,hostname --port 27017 --dbpath /data/mongodb/db1,/data/mongodb/db2,/data/mongodb/db3
4. 启动 Mongos
在 MongoDB 官方文档 (https://docs.mongodb.com/v4.4/tutorial/deploy-shard-cluster/) 中,提供了启动 Mongos 的脚本:
mongos --configdb configReplSet/hostname1:27019,hostname2:27019,hostname3:27019 --bind_ip localhost,hostname --port 27017
其中,--configdb 指定了 Config Server 的地址。如下是启动 Mongos 的示例命令:
mongos --configdb configReplSet/192.168.1.101:27019,192.168.1.102:27019,192.168.1.103:27019 --bind_ip 192.168.1.104 --port 27017
5. 添加 shard
在 Mongos 中添加 shard:
mongos> sh.addShard("shardReplSet1/192.168.1.105:27017")
其中,shardReplSet1 是 shard 集合名称,192.168.1.105:27017 是 shard 所在节点的地址。添加 shard 时,可以添加多个 shard。
6. 启用分片
分片集群搭建完成后,还需要启用分片。在 MongoDB 操作时,需要指定分片键,从而将数据分散到不同的 shard 上。下面的示例代码中使用了 name 作为分片键:
mongos> sh.enableSharding("test") mongos> sh.shardCollection("test.user", {"name": "hashed"})
其中,sh.enableSharding("test") 指定启用分片,并为数据库 test 开启分片功能。sh.shardCollection("test.user", {"name": "hashed"}) 指定了集合 user 采用分片策略,name 字段采用哈希分片策略。
实现原理
在 MongoDB 分片集群中,负责路由查询请求的是 Mongos,而 Config Server 存储整个集群的配置。对于一个查询请求来说,Mongos 首先需要通过 Config Server 获取 shard 的地址,然后将查询请求路由到对应的 shard 上。
查询流程
Mongos 接收到查询请求后,会向 Config Server 查询目标集合的 chunks,每个 chunk 都表示一个连续的区间。之后 Mongos 会将查询请求的 shard 键值根据 chunk 区间的范围进行定位,进而查询到对应的 shard 上。
写入流程
在 MongoDB 中,写入和查询的流程不同。在写入时,Mongos 首先通过 Config Server 获取目标集合的 chunks 信息,之后确定需要将数据写入到哪个 shard 上,因为每个 chunk 都会关联一个 primary shard(默认作为数据写入的 shard),读写该 chunk 的所有操作都发生在 primary shard 上。Mongos 根据目标 chunk 的 primary shard 将数据写入到对应的 shard 上,并在 chunk 区间上更新元数据,使得数据可以在整个集群内可见。
指导意义
通过本文,我们介绍了 MongoDB 分片集群的搭建方法及其实现原理,它可以应用于存储大规模数据的场景,并可以扩展性地支持更多数据的增长。在实际应用中,还需要注意数据备份、集群的高可用性以及监控等问题。总结一下,MongoDB 分片集群提供了:
- 分布式存储能力,可以通过横向扩展节点数又提高扩展性;
- 支持大规模数据存储,满足企业级数据存储需求;
- 高可用性和容错能力,支持集群的自动容错和故障转移。
示例代码
查询数据
db.user.find({"name": "Tom"})
插入数据
db.user.insert({"name": "Tom", "age": 23})
更新数据
db.user.update({"name": "Tom"}, {"$set": {"age": 24}})
删除数据
db.user.remove({"name": "Tom"})
结语
本文介绍了 MongoDB 分片集群的搭建方法及其实现原理,以及示例代码,希望能够为读者提供参考和帮助。在实际应用中,需要针对实际业务需求进行适当的规划和调整。如果有不足或错误之处,还请读者批评指正。
来源:JavaScript中文网 ,转载请注明来源 https://www.javascriptcn.com/post/65a201efadd4f0e0ffa1598e