前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Mysql两千万数据优化及迁移

Mysql两千万数据优化及迁移

作者头像
用户5224393
发布2019-08-13 15:51:13
1.5K0
发布2019-08-13 15:51:13
举报
文章被收录于专栏:Java研发军团Java研发军团
代码语言:javascript
复制
作者:一杯甜酒 原文:https://blog.csdn.net/u012562943/article/details/52457572

引言

最近有一张2000W条记录的数据表需要优化和迁移。2000W数据对于MySQL来说很尴尬,因为合理的创建索引速度还是挺快的,再怎么优化速度也得不到多大提升。

不过这些数据有大量的冗余字段和错误信息,极不方便做统计和分析。所以我需要创建一张新表,把旧表中的数据一条一条取出来优化后放回新表;

一. 清除冗余数据,优化字段结构

2000W数据中,能作为查询条件的字段我们是预知的。所以将这部分数据单独创建新的字段,对于有规则的数据合理改变字段结构,比如身份证就是varchar(18)。对于不重要的数据我们合并后存在一个结构为text的字段。

对于一些有关联的数据我们需要计算,常见的比如身份证种能获取到准确的性别,出生地、生日、年龄。

二. 数据迁移

我们从数据库中取出一条旧数据,再通过计算处理后得到想要的新数据,最后将新数据插入新表。不过在获取新数据时遇到如下问题。

1.数据量太大,无法一次获取(2000W数据扔到内存挺可怕的)

我们可以通过MySQL的limit语法分批获取。比如每次获取50000,SQL语句如下:

代码语言:javascript
复制
select * from table_name limit 15000000,50000;

通过这种方法能解决数据量太大的问题,但是随着limit的第一个参数越来越大,查询速度会慢的吓人(上面这条SQL执行会花35秒)。

时间就是生命,于是我们开始优化SQL语句,优化后变成下面这样:

代码语言:javascript
复制
select * from table_name order by id desc limit 5000000,50000;

可通过二分法拆分2000W数据,当执行到1000W数据时,将数据倒序。优化后SQL执行效率显著提升,从35秒降到9秒;

不过还是很慢,时间就是生命……还好我们有自增ID(创建数据表第一条定律,一定要有自增字段),优化后的SQl如下:

代码语言:javascript
复制
1. select * from table_name where id>15000000 and id<15050000;
2. select * from table_name where id>15000000 limit 50000;

为了直观演示,我写了两条功能一样的SQL。相比第一条,第二条的limit会导致SQL的索引命中变差,效率同样也会下降。

第一条SQL的执行时间是2毫秒,第二条执行时间5毫秒(我取的平均值)。每次数据的查询速度直接从35秒降到2毫秒……

2.数据量太大并且数据无法预估,某些特殊数据会导致数据导入失败;

我们有三种方案去将新数据存入新表,分别如下:

2.1一条一条插入数据;

开始肯定会想这种方案一定不行,因为每次插入都会有一次数据库IO操作。但是该方案有个好处是能及时发现有问题的数据,修改后再继续执行;

在Oracle中使用『绑定变量』能带来性能提升,正好MySQL也提供了『绑定变量』的功能。于是在不改变逻辑的情况下,尝试优化数据存储速度。代码如下:

代码语言:javascript
复制
public function actionTest(array $data)
{
   $mysqli = new mysqli("192.168.1.106", "username", "password", "test");
   $sql = "insert into table_name(name,identity) values (?,?)";
   $stmt = $connection->prepare($sql);
   $name = "";
   $identity = "";
   //使用绑定变量
   $stmt->bind_param("si", $name, $identity);
   foreach($data as $val)
   {
      $name = $val[name];
      $identity = $val[card_id];
      //执行
      $stmt->execute();
   }
   $stmt->close();
}

最后效果不怎么好,MySQL的『绑定变量』并没带来明显的速度提升,不过能有效的防止SQL注入;

2.2 一次插入50000条数据;

这是我最后选中的方案,一是能及时发现有问题的数据,二是导入数据非常稳定。就像支持断点续传一样,每一步都能看到效果。在执行脚本时,也能同步开始写分析逻辑;

2.3组装成SQL文件,最后统一导入;

组装一个大的SQL文件,最后通过MySQL自带的工具导入也是极好的。但如果有一条SQL有问题,你可能需要重跑一次脚本。因为在9G大小的文本文件中修改一个符号是很痛苦的事情……

三. 总结

通过各种优化,最后将脚本执行时间缩短到了20分钟内。优化后数据质量得到了较高保证,下次将尝试2亿数据的优化&迁移……

文章可以多多留言,小编会尽量搜集,谢谢!!!

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2018-10-23,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java研发军团 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 引言
  • 一. 清除冗余数据,优化字段结构
  • 二. 数据迁移
  • 三. 总结
相关产品与服务
云数据库 SQL Server
腾讯云数据库 SQL Server (TencentDB for SQL Server)是业界最常用的商用数据库之一,对基于 Windows 架构的应用程序具有完美的支持。TencentDB for SQL Server 拥有微软正版授权,可持续为用户提供最新的功能,避免未授权使用软件的风险。具有即开即用、稳定可靠、安全运行、弹性扩缩等特点。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档