在 MySQL 中查询分片数据

Posted

技术标签:

【中文标题】在 MySQL 中查询分片数据【英文标题】:Querying sharded data in MySQL 【发布时间】:2011-09-08 10:44:06 【问题描述】:

我正在处理 mysql 数据库中的大量数据,我想使用分片来横向扩展。我了解分片的原理,我什至知道我想如何分片我的数据。

当我查找数据库分片时,我找不到任何关于如何实际管理和查询分片数据库的全面示例。

具体来说,假设我已将数据拆分为多个表/数据库(分片),那么查询该数据的最佳方法是什么?我认为没有办法让 mysql 智能地知道使用哪个分片。

是否有第三方软件可以管理分片和我的查询?还是我必须更改我的代码(用 php 编写)才能与分片数据交互?

【问题讨论】:

【参考方案1】:

值得一提的是,我已经处理了一些更大的系统,并且有一个自定义的内部应用程序可以汇总来自服务器的查询,以用于公司的一般应用程序。

例如select * from t1 被转换为:

select * from db1.t1
union
select * from db2.t2

等等

主要问题是,如果您遇到跨服务器连接,在大型百万行以上的系统上,它会严重影响网络并且需要很长时间来处理查询。

假设您正在进行网络分析,需要对表进行连接以确定用户属性的“链接”。

您最终可能会遇到一些奇怪的查询,例如(请原谅语法):

  select db1.user1.boss, db1.user1.name, db2.user.name db2.user.boss from db1 inner join on db1.user.name = db2.user.name

(例如,找一个人的老板,以及他们的老板,或朋友的朋友等)

当您想要获得良好的数据来执行链接类型的查询时,这可能是一个巨大的 PITA,但是对于简单的统计数据,如总和、平均值等......最适合这些人的是每晚查询汇总统计数据进入每个服务器上的表(例如 nightlystats).. 例如select countif(user.datecreated>yesterday,1,0) as dailyregistered, sumif(user.quitdate)... into (the new nightly record).

这使得每日统计数据变得非常简单,因为您只需将总列相加,将单个服务器值乘以服务器总计数然后除以总总数等的平均值,并有一个非常快速的仪表板高层次的观点。

我们最终做了很多索引和优化,而保留常用信息的小型本地表等技巧有助于加快查询速度。

对于较大的查询,数据库人员只是将完整的系统副本转储到备份系统上,我们会在白天使用它在本地处理它,以免对网络造成太大影响。

有一些技巧可以减少这种情况,例如共享小表(例如用户的主表等不变的数据等),这样您就不必浪费时间收集这些数据。

在实践中真正有用的另一件事是将简单查询的总和和总计汇总到每晚的表格中。

最后一件令人感兴趣的事情是,bw 问题的解决方法是将“退避”超时编程到内部“查询聚合器”中,它所做的是记录获取响应的时间,如果时间开始延迟,它会要求更少的记录并增加它所要求的查询的延迟(因为它正在报告并且对时间不敏感,这很好用)

有一些 SQL 可以自动缩放,我最近阅读了一些关于工具(但不是 php)的文章,它们将为您完成其中的一些工作。我认为它们与云虚拟机提供商有关。

这个帖子也提供了一些工具和想法:MySQL sharding approaches?

如果 NoSQL 是一个选项,您可以考虑在走这条路之前查看所有的数据库系统。

不过,NoSQL 方法可能更容易扩展,具体取决于您要查找的内容。

【讨论】:

【参考方案2】:

使用Shard-Query。

见:http://www.mysqlperformanceblog.com/2011/05/14/distributed-set-processing-with-shard-query/

【讨论】:

文档不是很好,我无法让分片查询工作。马上,我从下载的文件中收到有关类的语法错误和错误。有没有替代品或好的演示? 如您所知,Shard-Query 得到了定期改进。它现在更易于使用和配置。我们正在努力完成我们的测试套件,以期待 2.0 版本的发布,这几乎是一个完整的重写,大大提高了 SQL 覆盖率。【参考方案3】:

您可以在 mysql 中使用分区或分片。如果您使用分区,那么 mysql 将根据 where 子句中的条件为您获取正确的数据。如果您使用分片,那么您需要定义一个分片键。因此,数据将根据分片键分片到表中。

假设你有一张employee表,根据employee_id对这张表进行分片,分片数为10。现在分片表中的数据可以放在表名中,比如employee_(employee_id % 10)。因此,员工数据将根据分片键进入名称为employee_1,employees_2 .....employees_10 的表中。

这里mysql不会自动计算表名,但你必须用你使用的语言来做。

【讨论】:

以上是关于在 MySQL 中查询分片数据的主要内容,如果未能解决你的问题,请参考以下文章

分片分区复制的区别及实现(mongodbmysql)

使用分层数据结构对数据库进行分片

JAVAEE——宜立方商城13:Mycat数据库分片主从复制读写分离100%Linux中成功安装Mysql的方法

mysql分片

ElasticSearch中index的设计

MyCat 学习笔记 第十一篇.数据分片 之 分片数据查询 ( select * from table_name limit 100000,100 )