【mysql】批量更新数据

摘要:
概述:批量更新MySQL数据表数据。在线搜索基本上有4-5种方法。我使用的更新方法是INSERT…ONDUPLICATEKEYUPDATE语法。参见官方网站:insert on duplicate函数:这意味着在插入时,如果主键或唯一索引重复,则不会执行插入操作,但会执行更新操作;注意:这种批量更新方法不是SQL规范,而是MySQL独有的;仅用于唯一索引(UNIQUEin
概述

批量更新mysql数据表数据,上网搜索基本都会说4~5方法,本人使用的更新方式为:
INSERT ... ON DUPLICATE KEY UPDATE Syntax
可参见官方网站:insert-on-duplicate

功能:

  • 表示插入时,如果遇到了主键重复唯一索引重复,则不执行插入操作,而是执行更新操作;

注意点:

  • 这种方式的批量更新,不是sql的规范,而是mysql特有的;
  • 只能针对唯一索引(UNIQUE index) 主键索引(RIMARY KEY)进行更新;
  • 对于自增主键,只会执行插入操作,不会进行更新;
  • 批量更新:values()方法很有用;

性能:

  • 对于数据量比较小的表,速度很快;
  • 对于数据量大的表,性能比较差,建议考虑其他方式;

如果使用Innodb引擎,则可以考虑如下方式(因为Innodb引擎支持事务)

START TRANSACTION;
UPDATE ...
UPDATE ...
UPDATE ...
UPDATE ...
COMMIT;

https://dba.stackexchange.com/questions/28282/whats-the-most-efficient-way-to-batch-update-queries-in-mysql

values(col_name)介绍

values(col_name):表示获取将要插入的列的值,注意是将要插入(would be inserted)


原始表结构和数据
CREATE TABLE `capacity_pm` (
  `id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自增主键',
  `pool_id` char(36) CHARACTER SET utf8 DEFAULT NULL COMMENT '资源池ID',
  `cluster_lv1` varchar(255) CHARACTER SET utf8 DEFAULT NULL COMMENT '集群分类',
  `cluster_lv2` varchar(255) CHARACTER SET utf8 DEFAULT NULL COMMENT '集群2级分类',
  `update_at` datetime DEFAULT CURRENT_TIMESTAMP COMMENT '更新或创建时间',
  `templete_id` varchar(255) CHARACTER SET utf8 NOT NULL COMMENT '模板ID',
  `templete_name` varchar(255) CHARACTER SET utf8 DEFAULT NULL COMMENT '模板名称',
  `templete_cpu_core` int(10) unsigned zerofill NOT NULL COMMENT '模板CPU核数',
  `templete_mem_size` double NOT NULL COMMENT '模板内存大小',
  `templete_disk_size` double NOT NULL COMMENT '模板磁盘大小',
  `host_total` int(11) unsigned zerofill DEFAULT NULL COMMENT '主机总数',
  `host_used` int(11) unsigned zerofill DEFAULT NULL COMMENT '主机已分配数量',
  `cpu_core_total` int(11) unsigned zerofill DEFAULT NULL COMMENT 'cpu总核数',
  `cpu_core_free` int(11) DEFAULT NULL,
  `cpu_core_used` int(11) DEFAULT NULL COMMENT 'cpu已分配数量',
  `cpu_core_util` double DEFAULT NULL COMMENT 'cpu核数使用占比',
  `mem_total` double DEFAULT NULL COMMENT '内存总空间',
  `mem_free` double DEFAULT NULL,
  `mem_used` double DEFAULT NULL,
  `mem_util` double DEFAULT NULL COMMENT '内存使用占比',
  `disk_total` double DEFAULT NULL,
  `disk_free` double DEFAULT NULL,
  `disk_used` double DEFAULT NULL,
  `disk_util` double DEFAULT NULL COMMENT '磁盘使用占比',
  PRIMARY KEY (`id`),
  UNIQUE KEY `idx_templete_all` (`pool_id`,`templete_id`) USING BTREE COMMENT '模块ID做完整索引'
) ENGINE=InnoDB AUTO_INCREMENT=70 DEFAULT CHARSET=utf8 COLLATE=utf8_bin;


INSERT INTO `capacity_pm` VALUES ('1', '7b8f0f5e2fbb4d9aa2d5fd55466d638f', null, null, '2018-04-11 15:04:31', 't001', '数据库服务器', '0000000000', '0', '0', '00000000100', '00000000010', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('2', '7b8f0f5e2fbb4d9aa2d5fd55466d638f', null, null, '2018-04-11 15:04:31', 't002', '性能性服务器', '0000000000', '0', '0', '00000000200', '00000000020', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('3', '7b8f0f5e2fbb4d9aa2d5fd55466d638f', null, null, '2018-04-11 15:04:31', 't003', '计算型服务器', '0000000000', '0', '0', '00000000300', '00000000030', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('4', '7b8f0f5e2fbb4d9aa2d5fd55466d638f', null, null, '2018-04-11 15:04:31', 't004', '存储型服务器', '0000000000', '0', '0', '00000000400', '00000000040', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('5', '7b8f0f5e2fbb4d9aa2d5fd55466d638f', null, null, '2018-04-11 15:04:31', 't005', '网络型服务器', '0000000000', '0', '0', '00000000500', '00000000050', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('6', '7b8f0f5e2fbb4d9aa2d5fd55466d638e', null, null, '2018-04-11 15:04:31', 't001', '数据库服务器', '0000000000', '0', '0', '00000001000', '00000000100', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('7', '7b8f0f5e2fbb4d9aa2d5fd55466d638e', null, null, '2018-04-11 15:04:31', 't002', '性能性服务器', '0000000000', '0', '0', '00000002000', '00000000200', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('8', '7b8f0f5e2fbb4d9aa2d5fd55466d638e', null, null, '2018-04-11 15:04:31', 't003', '计算型服务器', '0000000000', '0', '0', '00000003000', '00000000300', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('9', '7b8f0f5e2fbb4d9aa2d5fd55466d638e', null, null, '2018-04-11 15:04:31', 't004', '存储型服务器', '0000000000', '0', '0', '00000004000', '00000000400', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('10', '7b8f0f5e2fbb4d9aa2d5fd55466d638e', null, null, '2018-04-11 15:04:31', 't005', '网络型服务器', '0000000000', '0', '0', '00000005000', '00000000500', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('12', '7b8f0f5e2fbb4d9aa2d5fd55466d638e', null, null, '2018-04-11 08:12:00', 't006', '自定义服务器', '0000000000', '0', '0', '00000006000', '00000000600', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('13', '7b8f0f5e2fbb4d9aa2d5fd55466d638e', null, null, '2018-04-11 08:12:36', 't007', 'xxx服务器', '0000000000', '0', '0', '00000007000', '00000000700', null, null, null, null, null, null, null, null, null, null, null, null);
INSERT INTO `capacity_pm` VALUES ('14', '7b8f0f5e2fbb4d9aa2d5fd55466d638e', null, null, '2018-04-11 08:12:36', 't00x', '服务器xxx', '0000000000', '0', '0', '00000008000', '00000000800', null, null, null, null, null, null, null, null, null, null, null, null);

部分数据集查询如下:(该部分为重点测试的数据)

mysql> SELECT pool_id, templete_id, host_total, host_used from capacity_pm ;
+----------------------------------+-------------+------------+-----------+
| pool_id                          | templete_id | host_total | host_used |
+----------------------------------+-------------+------------+-----------+
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t001        |        100 |        10 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t002        |        200 |        20 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t003        |        300 |        30 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t004        |        400 |        40 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t005        |        500 |        50 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t001        |       1000 |       100 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t002        |       2000 |       200 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t003        |       3000 |       300 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t004        |       4000 |       400 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t005        |       5000 |       500 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t006        |       6000 |       600 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t007        |       7000 |       700 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t00x        |       8000 |       800 |
+----------------------------------+-------------+------------+-----------+
插入新数据-测试
  • 自增主键:上表中,主键是自增主键,所以该种批量更新方式对自增主键无效(因为自增主键只会insert数据,并不会update);
  • 唯一索引:UNIQUE KEY idx_templete_all (pool_id,templete_id)

待插入的数据,和表中的初始数据有唯一索引重复,索引会执行update操作,而非insert操作;

插入语句为:(所有的host_used都有变化)

INSERT INTO `capacity_pm`(pool_id, templete_id, host_total, host_used) values
('7b8f0f5e2fbb4d9aa2d5fd55466d638f', 't001','100', '15'),
('7b8f0f5e2fbb4d9aa2d5fd55466d638f', 't002','200', '25'),
('7b8f0f5e2fbb4d9aa2d5fd55466d638f', 't003','300', '35'),
('7b8f0f5e2fbb4d9aa2d5fd55466d638f', 't004','400', '45'),
('7b8f0f5e2fbb4d9aa2d5fd55466d638f', 't005','500', '55') 
ON DUPLICATE KEY UPDATE host_total=VALUES(host_total), host_used=VALUES(host_used);

ON DUPLICATE KEY UPDATE host_total=VALUES(host_total), host_used=VALUES(host_used)

  • host_total=VALUES(host_total): values(col_name)表示待插入的记录的值;
  • host_used=VALUES(host_used):当需要更新多个col时,使用“,”分割;

插入结果: host_used都发生了变化

mysql> SELECT pool_id, templete_id, host_total, host_used from capacity_pm ;
+----------------------------------+-------------+------------+-----------+
| pool_id                          | templete_id | host_total | host_used |
+----------------------------------+-------------+------------+-----------+
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t001        |        100 |        15 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t002        |        200 |        25 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t003        |        300 |        35 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t004        |        400 |        45 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638f | t005        |        500 |        55 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t001        |       1000 |       100 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t002        |       2000 |       200 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t003        |       3000 |       300 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t004        |       4000 |       400 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t005        |       5000 |       500 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t006        |       6000 |       600 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t007        |       7000 |       700 |
| 7b8f0f5e2fbb4d9aa2d5fd55466d638e | t00x        |       8000 |       800 |
+----------------------------------+-------------+------------+-----------+
13 rows in set

其他注意点

唯一索引:ALL唯一索引字段都不能为空,否则无法达到update操作;


性能比较

批量更新5w条数据

INSERT ... ON DUPLICATE KEY UPDATE Syntax

自己的机器上运行,大约30s;

事务批量更新

START TRANSACTION;
UPDATE ...
UPDATE ...
UPDATE ...
UPDATE ...
COMMIT;

测试结果:耗时特别长,不知道具体原因

免责声明:文章转载自《【mysql】批量更新数据》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇axios 正确打开方式爬虫笔记之w3cschool注册页面滑块验证码破解(巨简单滑块位置识别,非鼠标模拟轨迹)下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

部署Qualitis

Qualitis是一个支持多种异构数据源的质量校验、通知、管理服务的数据质量管理平台,用于解决业务系统运行、数据中心建设及数据治理过程中的各种数据质量问题。 Qualitis基于Spring Boot,依赖于Linkis进行数据计算,提供数据质量模型构建,数据质量模型执行,数据质量任务管理,异常数据发现保存以及数据质量报表生成等功能。并提供了金融级数据质量...

【DB宝41】监控利器PMM的使用--监控MySQL、PG、MongoDB、ProxySQL等

目录 一、PMM简介 二、安装使用 三、监控MySQL数据库MySQL慢查询分析 四、监控PG数据库 五、监控MongoDB数据库 六、监控ProxySQL中间件 一、PMM简介 之前发布过一篇Prometheus+Grafana的文章,连接为:【DB宝36】使用Docker分分钟搭建漂亮的prometheus+grafana监控,今天我们...

windows上备份mysql数据库

方案一:采用mysql自带的工具mysqldump。 脚本文件backup.bat如下: set  "YMD=%date:~,4%%date:~5,2%%date:~8,2%"cd    /d   C:\Program Files\MySQL\MySQL Server 5.7\binmysqldump  --no-defaults    -uroot   ...

tp5 使用SMproxy连接池来连接mysql数据库

前言:之前面试的时候被问到tp5-mysql连接池的问题,一下就蒙了,不知道啥玩意,今天有时间特意百度了一下。 PHP连接池的设计初衷是PHP 没有连接池,所以高并发时数据库会出现连接打满的情况,Mycat 等数据库中间件会出现部分 SQL 无法使用,例如不支持批量添加等,而且过于臃肿。 smproxy的原理是将数据库连接作为对象存储在内存中,当用户需要访...

python之路——表操作

阅读目录   引擎介绍   表介绍   创建表   查看表结构   mysql中的数据类型   表的完整性约束   修改表结构   删除表   多表结构的创建与分析   作业 返回顶部 引擎介绍 mysql中的存储引擎(https://www.cnblogs.com/l-hf/p/11533999.html) 返回顶部 表介绍 表就相当于文...

mysql 安装失败 start service执行不下去

简单说一下自己安装mysql的经历坑点,新手应该都会遇到。新买了一个电脑,第一次安装的情况:在网上下载好几个不同的mysql,安装都在最后一步执行的时候,执行不下去,最后打开客户端,就是闪一下就消失了,始终没有解决,最后在公司经理考了一个回来,安装竟然成功了,我也没有删除什么注册表什么的。第二次安装的情况:我拿经理给我的mysql安装,怎么也按照不成功,还...