mysql如何快速插入一千万条数据_如何快速安全的插入千万条数据?

最近有个需求解析一个订单文件,并且说明文件可达到千万条数据,每条数据大概在20个字段左右,每个字段使用逗号分隔,需要尽量在半小时内入库。

思路

1.估算文件大小

因为告诉文件有千万条,同时每条记录大概在20个字段左右,所以可以大致估算一下整个订单文件的大小,方法也很简单使用FileWriter往文件中插入一千万条数据,查看文件大小,经测试大概在1.5G左右;

2.如何批量插入

由上可知文件比较大,一次性读取内存肯定不行,方法是每次从当前订单文件中截取一部分数据,然后进行批量插入,如何批次插入可以使用insert(...)values(...),(...)的方式,经测试这种方式效率还是挺高的;怎么快速插入 100 条数据,用时最短,这篇看下。

3.数据的完整性

截取数据的时候需要注意,需要保证数据的完整性,每条记录最后都是一个换行符,需要根据这个标识保证每次截取都是整条数,不要出现半条数据这种情况;

4.数据库是否支持批次数据

因为需要进行批次数据的插入,数据库是否支持大量数据写入,比如这边使用的mysql,可以通过设置max_allowed_packet来保证批次提交的数据量;

5.中途出错的情况

因为是大文件解析,如果中途出现错误,比如数据刚好插入到900w的时候,数据库连接失败,这种情况不可能重新来插一遍,所有需要记录每次插入数据的位置,并且需要保证和批次插入的数据在同一个事务中,这样恢复之后可以从记录的位置开始继续插入。

实现

1.准备数据表

这里需要准备两张表分别是:订单状态位置信息表,订单表;

CREATE TABLE `file_analysis` (

`id` bigint(20) NOT NULL AUTO_INCREMENT,

`file_type` varchar(255) NOT NULL COMMENT '文件类型 01:类型1,02:类型2',

`file_name` varchar(255) NOT NULL COMMENT '文件名称',

`file_path` varchar(255) NOT NULL COMMENT '文件路径',

`status` varchar(255) NOT NULL COMMENT '文件状态 0初始化;1成功;2失败:3处理中',

`position` bigint(20) NOT NULL COMMENT '上一次处理完成的位置',

`crt_time` datetime NOT NULL COMMENT '创建时间',

`upd_time` datetime NOT NULL COMMENT '更新时间',

PRIMARY KEY (`id`)

) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8

CREATE TABLE `file_order` (

`id` bigint(20) NOT NULL AUTO_INCREMENT,

`file_id` bigint(20) DEFAULT NULL,

`field1` varchar(255) DEFAULT NULL,

`field2` varchar(255) DEFAULT NULL,

`field3` varchar(255) DEFAULT NULL,

`field4` varchar(255) DEFAULT NULL,

`field5` varchar(255) DEFAULT NULL,

`field6` varchar(255) DEFAULT NULL,

`field7` varchar(255) DEFAULT NULL,

`field8` varchar(255) DEFAULT NULL,

`field9` varchar(255) DEFAULT NULL,

`field10` varchar(255) DEFAULT NULL,

`field11` varchar(255) DEFAULT NULL,

`field12` varchar(255) DEFAULT NULL,

`field13` varchar(255) DEFAULT NULL,

`field14` varchar(255) DEFAULT NULL,

`field15` varchar(255) DEFAULT NULL,

`field16` varchar(255) DEFAULT NULL,

`field17` varchar(255) DEFAULT NULL,

`field18` varchar(255) DEFAULT NULL,

`crt_time` datetime NOT NULL COMMENT '创建时间',

`upd_time` datetime NOT NULL COMMENT '更新时间',

PRIMARY KEY (`id`)

) ENGINE=InnoDB AUTO_INCREMENT=10000024 DEFAULT CHARSET=utf8

2.配置数据库包大小

mysql> show VARIABLES like '%max_allowed_packet%';

+--------------------------+------------+

| Variable_name | Value |

+--------------------------+------------+

| max_allowed_packet | 1048576 |

| slave_max_allowed_packet | 1073741824 |

+--------------------------+------------+

2 rows in set

mysql> set global max_allowed_packet = 1024*1024*10;

Query OK, 0 rows affected

通过设置max_allowed_packet,保证数据库能够接收批次插入的数据包大小;不然会出现如下错误:

Caused by: com.mysql.jdbc.PacketTooBigException: Packet for query is too large (4980577 > 1048576). You can change this value on the server by setting the max_allowed_packet' variable.

at com.mysql.jdbc.MysqlIO.send(MysqlIO.java:3915)

at com.mysql.jdbc.MysqlIO.sendCommand(MysqlIO.java:2598)

at com.mysql.jdbc.MysqlIO.sqlQueryDirect(MysqlIO.java:2778)

at com.mysql.jdbc.ConnectionImpl.execSQL(ConnectionImpl.java:2834)

3.准备测试数据

public static void main(String[] args) throws IOException {

FileWriter out = new FileWriter(new File("D://xxxxxxx//orders.txt"));

for (int i = 0; i

out.write(

"vaule1,vaule2,vaule3,vaule4,vaule5,vaule6,vaule7,vaule8,vaule9,vaule10,vaule11,vaule12,vaule13,vaule14,vaule15,vaule16,vaule17,vaule18");

out.write(System.getProperty("line.separator"));

}

out.close();

}

使用FileWriter遍历往一个文件里插入1000w条数据即可,这个速度还是很快的,不要忘了在每条数据的后面添加换行符(\n\r);

4.截取数据的完整性

除了需要设置每次读取文件的大小,同时还需要设置一个参数,用来每次获取一小部分数据,从这小部分数据中获取换行符(\n\r),如果获取不到一直累加直接获取为止,这个值设置大小大致同每条数据的大小差不多合适,部分实现如下:

ByteBuffer byteBuffer = ByteBuffer.allocate(buffSize); // 申请一个缓存区

long endPosition = batchFileSize + startPosition - buffSize;// 子文件结束位置

long startTime, endTime;

for (int i = 0; i < count; i++) {

startTime = System.currentTimeMillis();

if (i + 1 != count) {

int read = inputChannel.read(byteBuffer, endPosition);// 读取数据

readW: while (read != -1) {

byteBuffer.flip();// 切换读模式

byte[] array = byteBuffer.array();

for (int j = 0; j

byte b = array[j];

if (b == 10 || b == 13) { // 判断\n\r

endPosition += j;

break readW;

}

}

endPosition += buffSize;

byteBuffer.clear(); // 重置缓存块指针

read = inputChannel.read(byteBuffer, endPosition);

}

} else {

endPosition = fileSize; // 最后一个文件直接指向文件末尾

}

...省略,更多可以查看Github完整代码...

}

如上代码所示开辟了一个缓冲区,根据每行数据大小来定大概在200字节左右,然后通过遍历查找换行符(\n\r),找到以后将当前的位置加到之前的结束位置上,保证了数据的完整性;

5.批次插入数据

通过insert(...)values(...),(...)的方式批次插入数据,部分代码如下:

// 保存订单和解析位置保证在一个事务中

SqlSession session = sqlSessionFactory.openSession();

try {

long startTime = System.currentTimeMillis();

FielAnalysisMapper fielAnalysisMapper = session.getMapper(FielAnalysisMapper.class);

FileOrderMapper fileOrderMapper = session.getMapper(FileOrderMapper.class);

fileOrderMapper.batchInsert(orderList);

// 更新上次解析到的位置,同时指定更新时间

fileAnalysis.setPosition(endPosition + 1);

fileAnalysis.setStatus("3");

fileAnalysis.setUpdTime(new Date());

fielAnalysisMapper.updateFileAnalysis(fileAnalysis);

session.commit();

long endTime = System.currentTimeMillis();

System.out.println("===插入数据花费:" + (endTime - startTime) + "ms===");

} catch (Exception e) {

session.rollback();

} finally {

session.close();

}

...省略,更多可以查看Github完整代码...

如上代码在一个事务中同时保存批次订单数据和文件解析位置信息,batchInsert通过使用mybatis的标签来遍历订单列表,生成values数据;

总结

以上展示了部分代码,完整的代码可以查看Github地址中的batchInsert模块,本地设置每次截取的文件大小为2M。

经测试1000w条数据(大小1.5G左右)插入mysql数据库中,大概花费时间在20分钟左右,当然可以通过设置截取的文件大小,花费的时间也会相应的改变。

推荐去我的博客:

觉得不错,别忘了点赞+转发哦!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/291506.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

解决 同时安装 python3,python2环境时,用pip安装 python3 包

应用场景 默认mac上已经安装了 python2; 而我又安装了 python3&#xff0c;并使用 python3; 安装了 pip 默认&#xff0c;pip安装的包安装在了 python2上了&#xff1b; 但是我想用 pip把安装的包安装在 python3上 &#xff0c;所以如下解决方式&#xff1b; 1&#xff1a;在ma…

C/C++之#ifdef、#if、#if defined的区别

1、看代码 2、运行结果 3、分析 #fi&#xff1a;后面接的表达式&#xff0c;如果为1就编译包含里面的内容 #ifdef&#xff1a;后面接的是一个宏&#xff0c;只要定义这个宏就行 #if defined(x)&#xff1a;和#ifdef效果一样 #if !defined(x)&#xff1a;和#ifndef效果一样

如何下载EP的各个版本?

到teamCity上面去下载 http://adc00cbv.us.oracle.com:8090/ 这里面刚进去是什么都没有的&#xff0c;要点击 Configure visible projects 配置一下才会显示 EP的各个版本是在V6.1.1.X中去下载的&#xff0c;也即EP和prodika是在一起出release 版本的。 转载于:https://www.cnb…

mysql丢失链接_MySQL远程连接丢失问题解决方法

最近远程连接mysql总是提示Lost connection to MySQL server at ‘reading initial communication packet’, system error: 0很明显这是连接初始化阶段就丢失了连接的错误其实问题很简单&#xff0c;都是MySQL的配置文件默认没有为远程连接配置好&#xff0c;只需要更改下MySQ…

.Net Core 读取文件时中文乱码问题的解决方法

背景今天在使用core web api上传txt文档的时候本来很顺利的&#xff0c;但是一测试发现读取的中文内容是乱码的&#xff0c;很是纳闷。出于经验&#xff0c;立马把代码的Encoding.Default改成 Encoding.uft8, 发现还是不行。后面索性把上传的文件另存为下&#xff0c;特地选择带…

mysql表空间被占用,同名表无法创建或导入

删除mysql表&#xff08;用的是innodb&#xff09;时没有用drop table命令&#xff0c;只是简单删除表目录&#xff0c;这导致表空间还存在&#xff0c;这样就不可以加同名表进去。当要重新导入新的同名表或者创建新的同名表时&#xff0c;会提示错误Error : Tablespace for ta…

C++编译代码的时候提示‘getInstance’ is not a member of ‘A’ A a = A::getInstance();解决办法

今天搞C类模板的时候&#xff0c;写个实例&#xff0c;发现提示下面的错误 ‘getInstance’ is not a member of ‘A’A& a A::getInstance();代码&#xff1a; class A {public:static T& Getinstance() {//}};//调用的时候写成了A::getinstance(); 解决办法&#…

关于使用indexedDB的本地存储(2)

我又回来了~这几天估计没喝茶&#xff0c;每天头都晕晕的&#xff0c;昨晚上和室友看了素鸡7&#xff0c;伤心啊&#xff0c;自己一直都喜欢这个系列&#xff0c;感觉童年真的是渐行渐远了…… 上一篇说到了哪些内容我这里罗列一下 建立和打开数据库、删除数据库、判断objectSt…

mysql数据库时间突然是12小时制_为什么存入mysql数据库中的timestamp,晚了13或14个小时...

# 为什么存入mysql数据库中的timestamp&#xff0c;晚了13个小时## 查看数据库时区show variables like %time_zone%;select global.system_time_zone;select global.time_zone;可以得到默认数据库时区&#xff1a;system_time_zone | CST |time_zone | SYSTEM|## CST时区&…

BCVP开发者社区2022专属周边第一弹

BCVP TeamBCVP开发者社区是博主老张的哲学发起&#xff0c;鼓励每个人都可参与的一个分享社区&#xff0c;目前已经有12个参与者&#xff0c;19个开源项目。欢迎加入BCVP&#xff0c;获取专属周边礼品&#xff08;文末有介绍&#xff09;。官方博客还在筹建中&#xff0c;预计2…

转 php include

http://www.w3school.com.cn/php/php_includes.asp PHP include 实例 例子 1 假设我们有一个名为 "footer.php" 的标准的页脚文件&#xff0c;就像这样&#xff1a; <?php echo "<p>Copyright © 2006-" . date("Y") . " W3S…

在C++中调用DLL中的函数(2)

本文转自&#xff1a;http://blog.sina.com.cn/s/blog_53004b4901009h3b.html 应用程序使用DLL可以采用两种方式&#xff1a;一种是隐式链接&#xff0c;另一种是显式链接。在使用DLL之前首先要知道DLL中函数的结构信息。Visual C6.0在VC\bin目录下提供了一个名为Dumpbin.exe的…

C++之类模板最简单的使用

1、说明类模板 1) 声明类模板时要增加一行 template <class 类型参数名> template意思是“模板”,是声明类模板时必须写的关键字。在template后面的尖括号内的内容为模板的参数表列,关键字class表示其后面的是类型参数 2、写代码理解 3、运行结果 4、总结 上…

win7将 esc与 capslock 互换

一天手软&#xff0c;于是买了一个机械键盘。cherry g80-3494 红轴各方面都不错就是有一个问题我经常用vim&#xff0c;其中esc。及F键区离主键盘区实在是太远了。 习惯于vim模式的人都有一种懒症&#xff0c;就是手指非常的不喜欢就离开了主键盘区。 于是就寻思着怎么样解决这…

mysql 5.5主从同步_MySQL5.5+配置主从同步并结合ThinkPHP5设置分布式数据库

This browser does not support music or audio playback. Please play it in WeChat or another browser.前言&#xff1a;本文章是在同处局域网内的两台windows电脑&#xff0c;且MySQL是5.5以上版本下进行的一主多从同步配置&#xff0c;并且使用的是集成环境工具PHPStudy为…

C# 10的新特性

点击上方蓝字关注我们&#xff08;本文阅读所需15分钟&#xff09;我们很高兴地宣布 C# 10 作为 .NET 6 和 Visual Studio 2022的一部分已经发布了。在这篇文章中&#xff0c;我们将介绍 C# 10 的许多新功能&#xff0c;这些功能使您的代码更漂亮、更具表现力、更快。阅读 Visu…

多线程介绍和多线程模块-lock-互斥锁

多线程介绍和多线程模块线程的特点&#xff1a;线程的生命周期开始运行结束线程的退出&#xff1a;进程执行完成线程的退出方法python的系统推出模块函数start_new_thread(func, args) #(func,(name,i))allocate_lock()exit()[root133 managehosts]# vim thread01.py #!/usr/bi…

C++编译出现binding ‘const string {aka const std::__cxx11::basic_string<char>}’ to reference of type ‘std

编译异常如下&#xff1a; 解决办法&#xff1a; 我的函数是这样的 string &larger(const string &s1, const string &s2){return s1.size() > s2.size()? s1 : s2; }改成这样就行了 const string &larger(const string &s1, const string &s2){r…

POJ 3181 Dollar Dayz DP

f[i][j]f[i-j][j]f[i][j-1]&#xff0c;结果很大需要高精度。 //#pragma comment(linker, "/STACK:1024000000,1024000000") #include<cstdio> #include<cstring> #include<cstdlib> #include<algorithm> #include<iostream> #include…

JS获取本周、本季度、本月、上月的开端日期、停止日期

Js代码 /** * 获取本周、本季度、本月、上月的开端日期、停止日期 */ var now new Date(); //当前日期 var nowDayOfWeek now.getDay(); //今天本周的第几天 var nowDay now.getDate(); //当前日 var nowMonth now.getMonth(); //当前月 var nowYear now.getYea…