最近,有一位同事,咨询我mysql的一点问题, 具体来说, 是如何很快的将一个mysql导出的文件快速的导入到另外一个mysql数据库。我学习了很多mysql的知识, 使用的时间却并不是很多, 对于mysql导入这类问题,我更是头一次碰到。询问我的原因,我大致可以猜到,以前互相之间有过很多交流,可能觉得我学习还是很认真可靠的。

首先,我了解了一下大致的情况, (1)这个文件是从mysql导出的,文件是运维给的,具体如何生成的,他不知道,可以询问运维 (2)按照当前他写的代码来看, 每秒可以插入几百条数据 (3)按照他们的要求, 需要每秒插入三四千条数据。(4)他们需要插入的数据量达到几亿条数据, 当前有一个上千万的实验数据

根据我学习的知识,在《高性能mysql》上面有过描述, load data的速度,比插入数据库快得多, 所以,我先通过他们从运维那里获取了生成数据的代码,通过向他们寻求了大约几千条数据。运维生成数据的方式是:select ... into outfile, 根据mysql官方文档上的说明,正好可以通过load data加载回数据库,load data使用的fields和lines等参数, 可以通过运维给出的语句得到,测试一次,成功。在我的推荐下,我们先使用innodb引擎根据测试的结果显示,大约每秒1千多。这个,我是通过date; mysql -e "load data ..."; date; 来大致得到。考虑到innodb中存储带索引的数据,插入速度会随着数据量的增加而变慢,那么我们使用几千条数据测试的结果,应该超过1千多。我们又在目标电脑上进行测试,得到的结果基本一致,速度会略快,具体原因当时没有细查。向远程mysql导入数据,需要进行一定的配置,具体配置的说明,在这里省略。因为目标电脑有额外的用处,所以,我们决定现在本机电脑上进行测试。

进行了初步的准备工作之后,我决定获取更多的数据,这次我打算下拉10万条数据。可是我的那位同事,通过对csv文件进行剪切,得到的结果,不能在我的mysql上进行很好的加载(load data)。于是,我们打算从有上千万条数据的测试mysql服务器上进行拉取。可是SELECT ... INTO OUTFILE只会将文件下拉到本地,我们没有本地的权限,除非找运维。为此,我们换了一种思路,使用select语句,模拟select ... into outfile语句,具体做法如下:

(1) 将数据select到本地,这里需要注意一件事,order by后面应该跟一个索引,这样可以避免排序操作。我进行过测试,如果不显示制定order by index, 下拉所用的时间会明显增加。
mysql -h 192.168.9.10 -u root -p -e "SELECT * FROM edu.olog order by idd LIMIT 100000;" > /home/sun/data.txt
(2) 使用sed命令移除输出中的第一行
sed -i '1d' /home/sun/data.txt
(3) 对远程数据库表格调用show create table指令得到表格的创建指令。在本机mysql的一个数据库中调用上述的创建表格的指令,这里可能需要做修改引擎一类的操作, 例如使用MYISAM作为表格的引擎,这样对于我们的加载任务来说,load data的速度明显更快。然后,对生成的文件调用load data记载到本机数据库, 不过没有FIELDS, lines一类的后缀。
这样,我们就可以对本机表格调用select ... into outfile生成对应的csv文件。
我们对十万条数据进行了测试,测试的结果显示,加载速度大约每秒1千多,速度比之前几千条时略慢。
 
我们已经知道了目标,也知道了当前的状态,那么下一步就是优化。首先,我查找了一些资料,这里以mysql8.0为例。首先查看了load data的文档,具体网页为:https://dev.mysql.com/doc/refman/8.0/en/load-data.html。考虑到load data类似于insert,我又在优化的相关模块中进行查找,大的目录为:https://dev.mysql.com/doc/refman/8.0/en/optimization.html(优化),紧接着查找的网页是:https://dev.mysql.com/doc/refman/8.0/en/insert-optimization.html(插入优化), 根据这个网页的提示,最后查看的网页是:https://dev.mysql.com/doc/refman/8.0/en/optimizing-innodb-bulk-data-loading.html(优化innodb表格的加载大量数据)。根据这里面的提示,我对我自己构建的测试数据进行了测试,使用的优化方法如下:
(1)临时修改自动提交的方式:
  1. SET autocommit=0;
  2. ... SQL import statements ...
  3. COMMIT;

(2)临时取消unique索引的检查:

  1. SET unique_checks=0;
  2. ... SQL import statements ...
  3. SET unique_checks=1;

这里简单介绍一下,我的测试使用表格和数据,我的create table如下:

  1. CREATE TABLE `tick` (
  2. `id` int(11) NOT NULL,
  3. `val` int(11) DEFAULT NULL,
  4. PRIMARY KEY (`id`)
  5. ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci

行数为1775232,插入时间大约在十几秒。在测试过程中,因为设计频繁的清空表格,清空表格需要花好几秒钟的时间,建议做类似操作的可以考虑,先drop table,然后再create table,这样速度会明显更快。按照在我的电脑上的测试结果,两者基本没有什么区别,大约都在18秒左右。使用的mysql语句为:

  1. date; mysql -u root -pmysql -e "load data infile './tick.txt' into table test.tick;";date;
  2.  
  3. mysql -u root -pmysql -e "set autocommit=0;set unique_checks=0;load data infile '/home/sun/tick.txt' into table test.tick;set unique_checks=1;set autocommit=1;";date;

为了确保我的设置生效,我检查了status输出:

  1. mysql -u root -pmysql -e "set autocommit=0;set unique_checks=0; show variables like '%commit';set unique_checks=1;set autocommit=1;";date;
  2.  
  3. date; mysql -u root -pmysql -e "set autocommit=0;set unique_checks=0; show variables like 'unique_checks';set unique_checks=1;set autocommit=1;";date;

输出结果如下:

autocommit为OFF, unique_checks为OFF。

其实我还设置过很多配置,例如插入缓冲区大小一类的,其中比较有效的是innodb_flush_log_at_trx_commit设置为2,对于我们这个问题,这个设置是可以考虑的。

我一直有一个疑问,就是说,load data是否为多线程运行的,按照我以往使用mysql的经验,和《高性能mysql》对load data的论断(远比insert快),如果多线程加载,不至于速度如此之慢。后来,我尝试使用mysqlimport,经过查看mysqlimport上面指定的--use-threads为多线程读取文件,以及在使用mysqlimport加载时,CPU利用率最多只有百分之两百多一点,这个让我觉得很有可能是单线程执行的。通过在运行load data的同时,调用show full processlist,可以清楚的看到,load data是单线程运行的。结果如下:

  1. *************************** 1. row ***************************
         Id: 4
       User: event_scheduler
       Host: localhost
         db: NULL
    Command: Daemon
       Time: 2835
      State: Waiting on empty queue
       Info: NULL
    *************************** 2. row ***************************
         Id: 17
       User: root
       Host: localhost
         db: test
    Command: Query
       Time: 0
      State: starting
       Info: show full processlist
    *************************** 3. row ***************************
         Id: 39
       User: root
       Host: localhost
         db: NULL
    Command: Query
       Time: 13
      State: executing
       Info: load data infile '/home/sun/tick.txt' into table test.tick
    3 rows in set (0.00 sec)

由上可以看出,load data是单线程执行的。

我知道,使用多线程执行,会是一个很好的办法,只是,我还是有想法提高单线程的效率,我想到了set profiling. 通过在一个session中设置set profiling = 1; 可以查看一个语句详细的时间消耗。我使用了如下的mysql语句:

  1. set profiling=1;
  2. load data infile '/home/sun/tick.txt' into table test.tick;
  3. show profiles;
  4. show profile for query 1;

可以得到如下的结果:

+----------------------+-----------+
| Status               | Duration  |
+----------------------+-----------+
| starting             |  0.000102 |
| checking permissions |  0.000016 |
| Opening tables       |  0.001377 |
| System lock          |  0.000024 |
| executing            | 17.476604 |
| query end            |  0.233398 |
| closing tables       |  0.000045 |
| freeing items        |  0.000039 |
| cleaning up          |  0.000047 |
+----------------------+-----------+
可见执行时间占据了最长的时间,而执行时间应该主要是CPU密集型的(这个我没有足够的把握),从CPU的使用来看,这个论断应该是比较合理的。对于一些比较复杂的问题,我不建议这样分析,当然,这里也可以考虑不这样分析,可以借鉴《高性能mysql》中的做法,参考如下的脚本:

  1. #!/bin/sh
  2.  
  3. INTERVAL=
  4. PREFIX=$INTERVAL-sec-status
  5. RUNFILE=/home/sun/benchmarks/running
  6. mysql -e 'SHOW GLOBAL VARIABLES' >> mysql-variables
  7. while test -e $RUNFILE; do
  8. file=$(date +%F_%I)
  9. sleep=$(date +%s.%N | awk "{print $INTERVAL - (\$1 % $INTERVAL)}")
  10. sleep $sleep
  11. ts="$(date +"TS %s.%N %F %T")"
  12. loadavg="$(uptime)"
  13. echo "$ts $loadavg" >> $PREFIX-${file}-status
  14. mysql -e 'SHOW GLOBAL STATUS' >> $PREFIX-${file}-status &
  15. echo "$ts $loadavg" >> $PREFIX-${file}-innodbstatus
  16. mysql -e 'SHOW ENGINE INNODB STATUS\G' >> $PREFIX-${file}-innodbstatus &
  17. echo "$ts $loadavg" >> $PREFIX-${file}-processlist
  18. mysql -e 'SHOW FULL PROCESSLIST\G' >> $PREFIX-${file}-processlist &
  19. echo $ts
  20. done
  21. echo Exiting because $RUNFILE does not exist

调整中间的时间,打印出详细的信息,进行分析。我这里就没有做这件事。

下面考虑了多线程处理,查看select ... into outfile的文件,即tick.txt可知,文件中的每行对应于表中的一条数据,想把文件切分,可以使用wc -l得出行数,然后使用head与tail得到两个文件,然后对这两个文件再次使用head和tail,得到测试用的四个文件。进行这个测试,我写了很简单的两个shell脚本:

  1. #!/bin/bash
  2. date;mysql -u root -pmysql -e "load data infile '/home/sun/$1' into table test.tick;";date;
  3. exit ;
  4.  
  5. #!/bin/bash
  6. ./mysql-load.sh data1 &
  7. ./mysql-load.sh data2 &
  8. ./mysql-load.sh data3 &
  9. ./mysql-load.sh data4 &
  10. exit ;

通过测试,可以发现加载这些文件所用的时间由18秒降低到10秒。我还测试了加载两个文件,所用时间由18秒降低到12.5秒,可见多线程加载可以明显提高加载速度。因为我的电脑是4线程的,所以我决定使用真正的测试数据进行测试。

按照我们之间出现的一个小插曲,发现使用MYISAM的插入速度会明显快于InnoDB,使用MYISAM插入速度可以达到每秒八千左右。我的同事认为这个可以满足他的需求,所以就采用了MYISAM。那时,我的同事发现他的插入速度明显我之前测试的要快,希望我能找出原因,基于没有mysql没有进行什么特殊的配置,他的电脑配置和我的电脑配置基本一致,我觉得应该是表格的问题,可能性最大的就是引擎,后来发现他无意间使用了MYISAM作为存储引擎。关于两个电脑配置的差别,可以通过show variables将配置变量导入到文件中进行对比得到,这个属于后话。既然我的同事觉得那样可行,我就没有进一步测试多线程的效果,只是这个经历可以记录下来,留以后借鉴。

简述mysql问题处理的更多相关文章

  1. 一天五道Java面试题----第九天(简述MySQL中索引类型对数据库的性能的影响--------->缓存雪崩、缓存穿透、缓存击穿)

    这里是参考B站上的大佬做的面试题笔记.大家也可以去看视频讲解!!! 文章目录 1.简述MySQL中索引类型对数据库的性能的影响 2.RDB和AOF机制 3.Redis的过期键的删除策略 4.Redis ...

  2. 15、简述MySQL的执行计划?

    具体的Mysql的执行计划,请参考下面的链接: MySQL_执行计划详细说明

  3. 如何回答——请简述MySQL索引类型

    想必大家在被问到这个问题的时候,在网上总是能搜到不同的回答,却又各不相同.其实这些答案大部分都是正确的,只不过在阐述MySQL索引类型的时候从不同方面入手而已.这里归纳如下,具体的机制可以参考其他博文 ...

  4. 简述MySQL数据库中的Date,DateTime,TimeStamp和Time类型

    DATETIME类型 定义同时包含日期和时间信息的值时.MySQL检索并且以'YYYY-MM-DD HH:MM:SS'格式显示DATETIME值,支持的范围是'1000-01-01 00:00:00' ...

  5. 简述MySQL优化

    数据库的优化可以从四个方面来优化: 1.结构层: web服务器采用负载均衡服务器,mysql服务器采用主从复制,读写分离 2.储存层: 采用合适的存储引擎,采用三范式 3.设计层: 采用分区分表,索引 ...

  6. Mysql性能优化二

    接上一篇Mysql性能优化一 建立适当的索引 说起提高数据库性能,索引是最物美价廉的东西了.不用加内存,不用改程序,不用调sql,只要执行个正确的'create index',查询速度就可能提高百倍千 ...

  7. Mysql主从架构的复制原理及配置详解

    一.简述Mysql复制 Mysql复制是通过将mysql的某一台主机的数据复制到其他主机(slaves)上,并且在slaves上重新执行一遍来实现.主服务器每次数据操作都会将更新记录到二进制日志文件, ...

  8. mysql in 的两种使用方法

    简述MySQL 的in 的两种使用方法: 他们各自是在 in keyword后跟一张表(记录集).以及在in后面加上字符串集. 先讲后面跟着一张表的. 首先阐述三张表的结构: s(sno,sname. ...

  9. Linux云计算运维-MySQL

    0.建初心 优秀DBA的素质 1.人品,不做某些事情2.严谨,运行命令前深思熟虑,三思而后行,即使是依据select3.细心,严格按照步骤一步一步执行,减少出错4.心态,遇到灾难,首先要稳住,不慌张, ...

随机推荐

  1. ESP8266 SDK开发: 测试下诱人的程序

    前言 这一节测试一下诱人的程序 实现的功能,APP通过SmartConfig给Wi-Fi模块配网并绑定设备,然后通过MQTT远程控制开发板的继电器, APP显示ESP8266采集的温湿度数据. 简而言 ...

  2. 网络协议 9 - TCP协议(下)

    上次了解了 TCP 建立连接与断开连接的过程,我们发现,TCP 会通过各种“套路”来保证传输数据的安全.除此之外,我们还大概了解了 TCP 包头格式所对应解决的五个问题:顺序问题.丢包问题.连接维护. ...

  3. 第01组 Alpha冲刺(5/6)

    队名:007 组长博客: https://www.cnblogs.com/Linrrui/p/11901035.html 作业博客: https://edu.cnblogs.com/campus/fz ...

  4. Solr7.x学习(5)-基本操作

    1.删除所有数据 在Documents中执行操作.Document Type选择XML:Document(s)输入:<delete><query>*:*</query&g ...

  5. swig包里面没有找到swig.exe

    问题关键词: swig.exe找不到 swig.exe不存在 windows如何编译swig.exe windows如何生成swig.exe SWIG简单介绍: SWIG(http://www.swi ...

  6. [转帖]String、StringBuilder与StringBuffer

    String.StringBuilder与StringBuffer https://www.jianshu.com/p/37f3799bdb56 1.String String本质 String是不可 ...

  7. Sitecore 内容版本设计

    Sitecore内容变化的跟踪显着偏离既定规范.了解Sitecore中版本控制和工作流程的细节,该产品是对这些发布工具的回答. 在出版界,实时跟踪内容变化很常见,可能是由于Microsoft Word ...

  8. TYPORA语法

    原文链接:https://blog.csdn.net/SIMBA1949/article/details/79001226

  9. sql中筛选第一条记录【分组排序】

    问题描述 我们现在有一张表titles,共有4个字段,分别是emp_no(员工编号),title(职位),from_date(起始时间),to_date(结束时间),记录的是员工在某个时间段内职位名称 ...

  10. 【题解】Luogu P5470 [NOI2019]序列

    原题传送门 同步赛上我一开始想了个看似正确却漏洞百出的贪心:按\(a_i+b_i\)的和从大向小贪心 随便想想发现是假的,然后就写了个28pts的暴力dp 杜神后半程说这题就是个贪心,但我没时间写了 ...