mysql replace语句
语法
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
REPLACE [LOW_PRIORITY | DELAYED] [ INTO ] tbl_name [PARTITION (partition_name,...)] [(col_name,...)] { VALUES | VALUE} ({expr | DEFAULT },...),(...),... Or : REPLACE [LOW_PRIORITY | DELAYED] [ INTO ] tbl_name [PARTITION (partition_name,...)] SET col_name={expr | DEFAULT }, ... Or : REPLACE [LOW_PRIORITY | DELAYED] [ INTO ] tbl_name [PARTITION (partition_name,...)] [(col_name,...)] SELECT ... |
原理
replace的工作机制有点像insert,只不过如果在表里如果一行有PRIMARY KEY或者UNIQUE索引,那么就会把老行删除然后插入新行。如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
root@test 03:23:55>show create table lingluo\G *************************** 1. row *************************** Table : lingluo Create Table : CREATE TABLE `lingluo` ( `a` int (11) NOT NULL DEFAULT '0' , `b` int (11) DEFAULT NULL , `c` int (11) DEFAULT NULL , `d` int (11) DEFAULT NULL , PRIMARY KEY (`a`), --------------------------同时存在PK约束 UNIQUE KEY `uk_bc` (`b`,`c`) ----------------唯一索引约束 ) ENGINE=InnoDB DEFAULT CHARSET=gbk 1 row in set (0.01 sec) root@test 02:01:44> select * from lingluo; Empty set (0.00 sec) root@test 03:27:40> replace into lingluo values (1,10000,3,4); --------表里没有已存在的记录相当于insert Query OK, 1 row affected (0.00 sec) -----------------------affect_rows是1 binlog格式: |
1
2
3
|
root@test 02:11:18> replace into lingluo values (1,10000,3,5); -------已经存在记录,且PK和UK同时冲突的时候,相当于先delete再insert Query OK, 2 rows affected (0.00 sec) ----------------------affect_rows是2,是delete和insert行数的总和 binlog格式: |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
root@test 02:26:09> select * from lingluo; + ---+-------+------+------+ | a | b | c | d | + ---+-------+------+------+ | 1 | 10000 | 3 | 5 | + ---+-------+------+------+ 1 row in set (0.00 sec) root@test 02:31:54> replace into lingluo values (1,10000,4,5); -------已经存在记录,且PK同时冲突的时候,相当于先delete再insert Query OK, 2 rows affected (0.00 sec) ---------------------------------affect_rows是2,是delete和insert行数的总和 root@test 02:32:02> select * from lingluo; + ---+-------+------+------+ | a | b | c | d | + ---+-------+------+------+ | 1 | 10000 | 4 | 5 | + ---+-------+------+------+ binlog格式: |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
root@test 02:37:04> replace into lingluo values (4,10000,6,5); Query OK, 1 row affected (0.00 sec) root@test 02:37:59> replace into lingluo values (6,10000,6,5); -------已经存在记录,且UK同时冲突的时候,直接update Query OK, 2 rows affected (0.00 sec) ---------------------------------affect_rows是2 root@test 02:40:31> select * from lingluo; + ---+-------+------+------+ | a | b | c | d | + ---+-------+------+------+ | 1 | 10000 | 4 | 5 | | 3 | 10000 | 5 | 5 | | 6 | 10000 | 6 | 5 | + ---+-------+------+------+ 3 rows in set (0.00 sec) binlog格式: |
疑问:
既然uk冲突的时候是update,那么为什么affect_rows都是2呢?让我们从源码上分析看下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
指定列 replace : root@test 03:34:37> select * from u; + ----+------+------+ | id | age | d | + ----+------+------+ | 0 | 1 | 126 | | 1 | 0 | 1 | | 3 | 1 | 123 | | 4 | 1 | 127 | | 5 | 0 | 12 | | 7 | 2 | 129 | + ----+------+------+ 6 rows in set (0.00 sec) root@test 03:34:37> select * from u; + ----+------+------+ | id | age | d | + ----+------+------+ | 0 | 1 | 126 | | 1 | 0 | 1 | | 3 | 1 | 123 | | 4 | 1 | 127 | | 5 | 0 | 12 | | 7 | 2 | 129 | + ----+------+------+ 6 rows in set (0.00 sec) root@test 03:34:40> replace into u (age,d) values (0,130); Query OK, 2 rows affected, 1 warning (0.01 sec) root@test 03:40:39>show warnings; + ---------+------+-----------------------------------------+ | Level | Code | Message | + ---------+------+-----------------------------------------+ | Warning | 1364 | Field 'id' doesn't have a default value | + ---------+------+-----------------------------------------+ 1 row in set (0.00 sec) root@test 03:40:47> select * from u; + ----+------+------+ | id | age | d | + ----+------+------+ | 0 | 0 | 130 | -----------------因为id是parimary但是没有auto_creasement,由126变成130 | 1 | 0 | 1 | | 3 | 1 | 123 | | 4 | 1 | 127 | | 5 | 0 | 12 | | 7 | 2 | 129 | + ----+------+------+ 6 rows in set (0.00 sec) |
用的时候需要注意的是:
如果指定replace列的话,尽量写全,要不然没有输入值的列数据会被赋成默认值(因为是先delete在insert),就和普通的insert是一样的,所以如果你要执行replace语句的话是需要insert和delete权限的。
如果你需要执行
SET
,就相当于执行col_name
=col_name
+ 1
.col_name
= DEFAULT(col_name
) + 1replace语句如果不深入看的话,就和insert一样,执行完后没什么反应
例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
|
root@test 04:20:04> select * from u; + ----+------+------+ | id | age | d | + ----+------+------+ | 0 | 0 | 130 | | 1 | 0 | 1 | | 3 | 1 | 123 | | 4 | 1 | 127 | | 5 | 0 | 12 | | 7 | 2 | 129 | + ----+------+------+ 6 rows in set (0.00 sec) root@test 04:20:10> replace into u (id,d) values (8,232); Query OK, 1 row affected (0.01 sec) root@test 04:20:39> select * from u; + ----+------+------+ | id | age | d | + ----+------+------+ | 0 | 0 | 130 | | 1 | 0 | 1 | | 3 | 1 | 123 | | 4 | 1 | 127 | | 5 | 0 | 12 | | 7 | 2 | 129 | | 8 | NULL | 232 | + ----+------+------+ 7 rows in set (0.00 sec) root@test 04:20:43> replace into u (id,d) values (7,232); Query OK, 3 rows affected (0.01 sec) ----------注意这里affect_rows是3,因为主键7已经存在,唯一索引232已经存在,所以需要删除id为7和8的行,然后插入新行 root@test 04:20:52> select * from u; + ----+------+------+ | id | age | d | + ----+------+------+ | 0 | 0 | 130 | | 1 | 0 | 1 | | 3 | 1 | 123 | | 4 | 1 | 127 | | 5 | 0 | 12 | | 7 | NULL | 232 | + ----+------+------+ 6 rows in set (0.00 sec) root@test 04:20:55> |
MySQL给replace和load data....replace用的算法是:
尝试向表里插入新行
当表里唯一索引或者primary key冲突的时候:
a. delete冲突行
b.往表里再次插入新行
如果遇到重复行冲突,存储过程很可能当作update执行,而不是delete+insert,但是显式上都是一样的。这里没有用户可见的影响除了存储引擎层Handler_xxx
的状态变量。
因为REPLACE ... SELECT语句的结果依赖于select的行的顺序,但是顺序没办法保证都是一样的,有可能从master和slave的都不一样。正是基于这个原因,MySQL 5.6.4以后,REPLACE ... SELECT语句被标记为基于statement的复制模式不安全的。基于这个变化,当使用STATEMENT记录二进制日志的时候,如果有这样的语句就会在log里面输出一个告警,同样当使用MIXED行复制模式也会记录告警。
在MySQL5.6.6之前的版本,replace影响分区表就像MyISAM使用表级锁锁住所有的分区表一样。当使用 REPLACE ... PARTITION
语句时确实会发生上述情况。(使用基于行锁的InnoDB引起不会发生这种情况。)在MySQL 5.6.6以后的版本MySQL使用分区锁,只有当分区(只要没有分区表的列更新)包含了REPLACE语句并且WHERE实际匹配到的才会锁住那个分区;否则的话就会锁住整个表。
操作形式:
binlog格式:
结论
当存在pk冲突的时候是先delete再insert
当存在uk冲突的时候是直接update
mysql replace语句的更多相关文章
- Mysql Replace语句的使用
Mysql Replace语句的语法: REPLACE [LOW_PRIORITY | DELAYED] [INTO] tbl_name [(col_name,...)] VALUES ({expr ...
- 慎用MySQL replace语句
语法: REPLACE [LOW_PRIORITY | DELAYED] [INTO] tbl_name [PARTITION (partition_name,...)] [(col_name,... ...
- MySQL数据库INSERT、UPDATE、DELETE以及REPLACE语句的用法详解
本篇文章是对MySQL数据库INSERT.UPDATE.DELETE以及REPLACE语句的用法进行了详细的分析介绍,需要的朋友参考下 MySQL数据库insert和update语句引:用于操作数 ...
- MySQL replace函数替换字符串语句的用法(mysql字符串替换)
MySQL replace函数我们经常用到,下面就为您详细介绍MySQL replace函数的用法,希望对您学习MySQL replace函数方面能有所启迪. 最近在研究CMS,在数据转换的时候需要用 ...
- MySQL中的replace语句
一.背景 当使用replace语句更新access_apps表时,原有的mark列.remark列的信息丢失. CREATE TABLE `access_apps` ( `base` varcha ...
- 转载:MySQL数据库INSERT、UPDATE、DELETE以及REPLACE语句的用法详解
转自:http://www.jb51.net/article/39199.htm 本篇文章是对MySQL数据库INSERT.UPDATE.DELETE以及REPLACE语句的用法进行了详细的分析介绍, ...
- MySQL 当记录不存在时insert,当记录存在时update(ON DUPLICATE KEY UPDATE, REPLACE语句)
MySQL 当记录不存在时insert,当记录存在时更新 网上基本有三种解决方法. 第一种:示例一:insert多条记录 假设有一个主键为 client_id 的 clients 表,可以使用下面的语 ...
- MySQL replace into 使用详解 及 注意事项
REPLACE的运行与INSERT很相似.只有一点例外,假如表中的一个旧记录与一个用于PRIMARY KEY或一个UNIQUE索引的新记录具有相同的值,则在新记录被插入之前,旧记录被删除.注意:除非表 ...
- MySQL replace into 说明(insert into 增强版)
MySQL replace into 说明(insert into 增强版) 在插入数据到一个表时,通常是这种情况:1. 先推断数据是否存在: 2. 假设不存在,则插入:3.假设存在,则更新. 在 S ...
随机推荐
- C# winform 窗体应用程序之图片上传Oracle数据库保存字段BLOB
C# winform 窗体应用程序之图片上传Oracle数据库保存字段BLOB 我用的数据库是Oracle,就目前来看,许多数据库现在都倾向于Oracle数据库,对ORACLE数据库基本的操作也是必须 ...
- 【ZOJ2314】Reactor Cooling(有上下界的网络流)
前言 话说有上下界的网络流好像全机房就我一个人会手动滑稽,当然这是不可能的 Solution 其实这道题目就是一道板子题,主要讲解一下怎么做无源无汇的上下界最大流: 算法步骤 1.将每条边转换成0~u ...
- Day 1. 占位符的使用方法(%d,%s)(格式化输出)
方法1 name = input("请输入名字1") age = input ("请输入年龄") hobby = input("请输入兴趣爱好&qu ...
- Delphi - 10.1编译OSX10.12程序遇到错误解决了!
昨天,尝试Delphi的跨平台开发功能,在windows10下,做了一个控制台程序,发布目标平台是OSX10.12,中间配置过程都非常顺利,没有任何错误,但是当编译运行时候出现下面错误: [dccos ...
- Java 8 LocalDateTime 初使用
LocalTime : 只包括时间 LocalDate : 只包括日期 LocalDateTime : 包括日期和时间 JDBC映射 LocalTime 对应 time LocalDate 对应 d ...
- 数组序列化serialize
1,数据在网络中是以字符串形式传输,这样如果传输的是数组,首先将数组内容拼接成字符串进行发送,接收方拿到字符串,没法将其还原为数组.因此在网络传输的时候,为了保证数据类型的不丢失,先序列化,再发送. ...
- windows系统上安装与使用Android NDK r5
windows系统上安装与使用Android NDK r5 很早就听说了android的NDK应用,只是一直没有时间去研究,今天花了点时间在windows平台搭建了NDK环境,并成功运行了第一个简单 ...
- docker私服registry管理镜像
前言 首先试想这样一个场景:当在自己的机器上(docker中)构建了mysql镜像,eureka镜像等等微服务镜像,这些镜像有可能需要放到其他的机器上docker环境中去运行,实行分布式架构部署.但如 ...
- shell脚本实现无密码交互的SSH自动登陆
ssh连接远程主机时候询问密码,跟su.sudo命令的默认行为一样,是不从stdin读入数据的,据称是为安全考虑,但是有时候在脚本当中确实需要无人守值的登陆. 搜索一下不难找到类似的例子,使用expe ...
- 如何修复“sshd error: could not load host key”
问题:当我尝试SSH到一台远程服务器时,SSH客户端登陆失败并提示“Connection closed by X.X.X.X”.在SSH服务器那端,我看到这样的错误消息:“sshd error: co ...