在MySQL数据库中,如果在insert语句前面带上ON DUPLICATE KEY UPDATE 子句,而要插入的行与表中现有记录的惟一索引或主键中产生反复值,那么就会产生旧行的更新;如果插入的行数据与现有表中记录的惟一索引或者主键不反复,则执行新纪录插入操作。另外,ON DUPLICATE KEY UPDATE不能写where条件。
create table kid_score(id tinyint unsigned not null,birth_day date not null,score int unsigned not null,primary key(id, birth_day) --惟一索引是由 id + birth_day 两个字段组成) engine = InnoDB;--初始化数据insert into kid_score(id, birth_day, score) values (1,'2019-01-15',10),(2,'2019-01-16',20);insert into kid_score(id, birth_day, score) values (1,'2019-01-15',30) ON DUPLICATE KEY UPDATE score = score + 50;
分惟一索引反复、惟一索引不反复的状况。
惟一索引反复,那就是更新操作。
惟一索引不反复,就是插入操作。
须要留神的是:如果行作为新记录被插入,则受影响行的值为1;如果原有的记录被更新,则受影响行的值为2,如果更新的数据和已有的数据截然不同,则受影响的行数是0。
mysql> select * from kid_score;+----+------------+-------+| id | birth_day | score |+----+------------+-------+| 1 | 2019-01-15 | 10 || 2 | 2019-01-16 | 20 |+----+------------+-------+2 rows in set-- 惟一索引反复,执行更新mysql> insert into kid_score(id, birth_day, score) values (1,'2019-01-15',30) ON DUPLICATE KEY UPDATE score = score + 50;Query OK, 2 rows affectedmysql> select * from kid_score;+----+------------+-------+| id | birth_day | score |+----+------------+-------+| 1 | 2019-01-15 | 60 || 2 | 2019-01-16 | 20 |+----+------------+-------+2 rows in set-- 惟一索引不反复,执行插入mysql> insert into kid_score(id, birth_day, score) values (2,'2019-01-15',30) ON DUPLICATE KEY UPDATE score = score + 50;Query OK, 1 row affectedmysql> select * from kid_score;+----+------------+-------+| id | birth_day | score |+----+------------+-------+| 1 | 2019-01-15 | 60 || 2 | 2019-01-15 | 30 || 2 | 2019-01-16 | 20 |+----+------------+-------+3 rows in set-- 惟一索引反复,应该执行更新,但更新值与原值雷同mysql> insert into kid_score(id, birth_day, score) values (2,'2019-01-16',20) ON DUPLICATE KEY UPDATE score = 20;Query OK, 0 rows affectedmysql> select * from kid_score;+----+------------+-------+| id | birth_day | score |+----+------------+-------+| 1 | 2019-01-15 | 60 || 2 | 2019-01-15 | 30 || 2 | 2019-01-16 | 20 |+----+------------+-------+3 rows in set
最常见的形式就是为字段设置主键或惟一索引,当插入反复数据时,抛出谬误,程序终止,但这会给后续解决带来麻烦,因而须要对插入语句做非凡解决,尽量避开或疏忽异样,上面我简略介绍一下,感兴趣的敌人能够尝试一下:
这里为了不便演示,我新建了一个user测试表,次要有id,username,sex,address这4个字段,其中主键为id(自增),同时对username字段设置了惟一索引:
01. insert ignore into
即插入数据时,如果数据存在,则疏忽此次插入,前提条件是插入的数据字段设置了主键或惟一索引,测试SQL语句如下,当插入本条数据时,MySQL数据库会首先检索已有数据(也就是idx_username索引),如果存在,则疏忽本次插入,如果不存在,则失常插入数据:
在这里插入图片形容
02. on duplicate key update
即插入数据时,如果数据存在,则执行更新操作,前提条件同上,也是插入的数据字段设置了主键或惟一索引,测试SQL语句如下,当插入本条记录时,MySQL数据库会首先检索已有数据(idx_username索引),如果存在,则执行update更新操作,如果不存在,则直接插入:
在这里插入图片形容
03. replace into
即插入数据时,如果数据存在,则删除再插入,前提条件同上,插入的数据字段须要设置主键或惟一索引,测试SQL语句如下,当插入本条记录时,MySQL数据库会首先检索已有数据(idx_username索引),如果存在,则先删除旧数据,而后再插入,如果不存在,则直接插入:
在这里插入图片形容
04. insert if not exists
即insert into … select … where not exist … ,这种形式适宜于插入的数据字段没有设置主键或惟一索引,当插入一条数据时,首先判断MySQL数据库中是否存在这条数据,如果不存在,则失常插入,如果存在,则疏忽:
在这里插入图片形容
目前,就分享这4种MySQL解决反复数据的形式吧,前3种形式适宜字段设置了主键或惟一索引,最初一种形式则没有此限度,只有你相熟一下应用过程,很快就能把握的,网上也有相干材料和教程,介绍的十分具体,感兴趣的话,能够搜一下。