百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程字典 > 正文

MYSQL REPLACE INTO解析(mysqlchange,repair,remove)

toyiye 2024-07-05 01:21 10 浏览 0 评论

实现机制

REPLACE的运行与INSERT很相像,但当旧记录与新记录发生唯一键冲突时,会在新记录被插入之前,将旧记录被删除:

  1. 尝试把新行插入到表中 ;
  2. 当因为对于主键或唯一关键字出现重复关键字错误而造成插入失败时,从表中删除含有重复关键字值的(所有)冲突行;
  3. 再次尝试把新行插入到表中 。

返回值:REPLACE语句会返回一个数值,用来指示受影响的行的数目。该数是被删除和被插入的行数的和。受影响的行数可以容易地确定是否REPLACE只添加了一行,或者是否REPLACE也替换了其它行:检查该数是否为1(添加)或更大(替换)。

存在问题

  • 数据字段丢失

如果指定replace列的话,尽量写全,要不然没有输入值的列数据会被赋成默认值(因为是先delete在insert),就和普通的insert是一样的,所以如果你要执行replace语句的话是需要insert和delete权限的。

如果你需要执行 SET col_name = col_name + 1,就相当于执行col_name = DEFAULT(col_name) + 1.

  • 主从不一致

导致主从不一致的原因由于以下两方面的原因导致:

  1. Innodb对auto_increment的处理机制:当语句是insert时,Innodb会对auto_increment进行递增(不论是否insert成功),而对update,delete,select等语句则不更新;
  2. 当REPLACE语句在主库执行时,如果先按照insert将记录插入数据表成功,那么在主从同步的binlog日志(binlog_format=row)中,记录的就是insert row event;否则,在主库上“先执行delete后执行insert”这两步操作在binlog中会被记录成了一条update row event。

由以上原因可知,当在主库中执行REPLACE语句出现惟一键冲突时,主库虽然首先执行的insert操作是失败的,但auto_increment还是会递增;但到了备库,在row格式下,由于只产生了一条update row event,从库无法知道主库是一个replace语句,而且insert还失败了, 所以auto_increment在备库上不会递增。从而同样出现主从切换后一段时间内新主库的插入操作在新的从库上因为主键(id)冲突而导致插入失败。

如果业务对主键有强依赖,在使用REPLACE语句时需要慎重,推荐使用INSERT ... ON DUPLICATE KEY UPDATE来解决业务需求。

  • 主键消耗过快

由于REPLACE对于唯一键冲突都采用先删除再插入的方式,导致主键消耗过快且主键不连续。

测试案例

表定义

CREATE TABLE `test` (

`id` bigint(20) NOT NULL AUTO_INCREMENT,

`uk` varchar(30) NOT NULL COMMENT 'uk',

`name` varchar(20) NOT NULL DEFAULT '',

PRIMARY KEY (`id`),

UNIQUE KEY `uk` (`uk`)

) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4;

mysql> replace into test (`uk`, `name`) values('a', 'a');--------表里没有已存在的记录相当于insert

Query OK, 1 row affected (0.01 sec)-----------------------affect_rows是1

查看表的AUTO_INCREMENT值

mysql> select * from test;

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

| id | uk | name |

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

| 1 | a | a |

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

1 row in set (0.01 sec)

mysql> show create table test\G

*************************** 1. row ***************************

Table: test

Create Table: CREATE TABLE `test` (

`id` bigint(20) NOT NULL AUTO_INCREMENT,

`uk` varchar(30) NOT NULL COMMENT 'uk',

`name` varchar(20) NOT NULL DEFAULT '',

PRIMARY KEY (`id`),

UNIQUE KEY `uk` (`uk`)

) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4

1 row in set (0.00 sec)

replace已经存在的记录

mysql> replace into test (`uk`, `name`) values('a', 'a');-------已经存在记录,且UK冲突的时候,相当于先delete再insert

Query OK, 2 rows affected (0.00 sec)----------affect_rows是2,是delete和insert行数的总和

查询得到主键一直在增长

mysql> select * from test;

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

| id | uk | name |

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

| 2 | a | a |

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

1 row in set (0.00 sec)

mysql> show create table test\G

*************************** 1. row ***************************

Table: test

Create Table: CREATE TABLE `test` (

`id` bigint(20) NOT NULL AUTO_INCREMENT,

`uk` varchar(30) NOT NULL COMMENT 'uk',

`name` varchar(20) NOT NULL DEFAULT '',

PRIMARY KEY (`id`),

UNIQUE KEY `uk` (`uk`)

) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4

1 row in set (0.00 sec)

数据字段缺失:

mysql> replace into test (`uk`) values('a');

Query OK, 2 rows affected (0.01 sec)

mysql> select * from test;

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

| id | uk | name |

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

| 3 | a | |

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

1 row in set (0.00 sec)

当replace语句中没有包含所有字段(例如上面的name字段),该字段被赋成默认值(先delete后insert),所以尽量写全所有字段,相当于是set column=default(column)。

相关推荐

如何用 coco 数据集训练 Detectron2 模型?

随着最新的Pythorc1.3版本的发布,下一代完全重写了它以前的目标检测框架,新的目标检测框架被称为Detectron2。本教程将通过使用自定义coco数据集训练实例分割模型,帮助你开始使...

CICD联动阿里云容器服务Kubernetes实践之Bamboo篇

本文档以构建一个Java软件项目并部署到阿里云容器服务的Kubernetes集群为例说明如何使用Bamboo在阿里云Kubernetes服务上运行RemoteAgents并在agents上...

Open3D-ML点云语义分割实验【RandLA-Net】

作为点云Open3D-ML实验的一部分,我撰写了文章解释如何使用Tensorflow和PyTorch支持安装此库。为了测试安装,我解释了如何运行一个简单的Python脚本来可视化名为...

清理系统不用第三方工具(系统自带清理软件效果好不?)

清理优化系统一定要借助于优化工具吗?其实,手动优化系统也没有那么神秘,掌握了方法和技巧,系统清理也是一件简单和随心的事。一方面要为每一个可能产生累赘的文件找到清理的方法,另一方面要寻找能够提高工作效率...

【信创】联想开先终端开机不显示grub界面的修改方法

原文链接:【信创】联想开先终端开机不显示grub界面的修改方法...

如意玲珑成熟度再提升,三大发行版支持教程来啦!

前期,我们已分别发布如意玲珑在deepinV23与UOSV20、openEuler24.03发行版的操作指南,本文,我们将为大家详细介绍Ubuntu24.04、Debian12、op...

118种常见的多媒体文件格式(英文简写)

MP4[?mpi?f??]-MPEG-4Part14(MPEG-4第14部分)AVI[e?vi??a?]-AudioVideoInterleave(音视频交错)MOV[m...

密码丢了急上火?码住7种console密码紧急恢复方式!

身为攻城狮的你,...

CSGO丨CS2的cfg指令代码分享(csgo自己的cfg在哪里?config文件位置在哪?)

?...

使用open SSL生成局域网IP地址证书

某些特殊情况下,用户内网访问多可文档管理系统时需要启用SSL传输加密功能,但只有IP,没有域名和证书。这种情况下多可提供了一种免费可行的方式,通过openSSL生成免费证书。此方法生成证书浏览器会提示...

Python中加载配置文件(python怎么加载程序包)

我们在做开发的时候经常要使用配置文件,那么配置文件的加载就需要我们提前考虑,再不使用任何框架的情况下,我们通常会有两种解决办法:完整加载将所有配置信息一次性写入单一配置文件.部分加载将常用配置信息写...

python开发项目,不得不了解的.cfg配置文件

安装软件时,经常会见到后缀为.cfg、.ini的文件,一般我们不用管,只要不删就行。因为这些是程序安装、运行时需要用到的配置文件。但对开发者来说,这种文件是怎么回事就必须搞清了。本文从.cfg文件的创...

瑞芯微RK3568鸿蒙开发板OpenHarmony系统修改cfg文件权限方法

本文适用OpenHarmony开源鸿蒙系统,本次使用的是开源鸿蒙主板,搭载瑞芯微RK3568芯片。深圳触觉智能专注研发生产OpenHarmony开源鸿蒙硬件,包括核心板、开发板、嵌入式主板,工控整机等...

Python9:图像风格迁移-使用阿里的接口

先不多说,直接上结果图。#!/usr/bin/envpython#coding=utf-8importosfromaliyunsdkcore.clientimportAcsClient...

Python带你打造个性化的图片文字识别

我们的目标:从CSV文件读取用户的文件信息,并将文件名称修改为姓名格式的中文名称,进行规范资料整理,从而实现快速对多个文件进行重命名。最终效果:将原来无规律的文件名重命名为以姓名为名称的文件。技术点:...

取消回复欢迎 发表评论:

请填写验证码