Mysql 存在则修改 不存在则新增的两种实现方法

假设有一张demo表,主键为id,唯一索引是code

create table demo
(
	id int auto_increment,
	name int null,
	gender int null,
	age int null,
	code int null,
	constraint demo_pk
		primary key (id)
);

create unique index demo_code_uindex
	on demo (code);

方法一 replace into

replace into 会根据唯一索引或主键进行判断,如果存在则覆盖写入字段,如果不存在则新增。
此方法有坑,如果主键是自增的,且通过唯一索引来进行操作时,主键会变更,该方法底层是先进性delete,在insert
如果有子表依赖的话不建议使用。

replace into 事例

REPLACE INTO demo(id, name, gender, age, code) VALUES (1,'1',1,1,1)

通过主键修改,此时没有任何问题,id还是1
image

当我们通过唯一索引code来更改。

REPLACE INTO demo(name, gender, age, code) VALUES ('1',1,1,1)

image
没执行一次 主键id都会自增。

方法二 on duplicate key

on duplicate key 如果遇到重复的唯一索引则会进行update,否则进行新增,没有replcae的坑,不会先进行delete 在进行 insert

on duplicate key 事例:

INSERT INTO demo(name, gender, age, code)
VALUES ('1',
       1,
       2,
       2)
ON DUPLICATE KEY UPDATE name   = values(name),
                       gender = values(gender),
                       age    = values(age),
                       code   = if(values(code) = 1,values(code), code)

image

无论执行多少次,主键值都是不会变的。

但是此方法也有坑,如果表中不止一个唯一索引的话,在特定版本的mysql中容易产生dead lock(死锁)

当mysql执行INSERT ON DUPLICATE KEY的 INSERT时,存储引擎会检查插入的行是否会产生重复键错误。如果是的话,它会将现有的
行返回给mysql,mysql会更新它并将其发送回存储引擎。当表具有多个唯一或主键时,此语句对存储引擎检查密钥的顺序非常敏感。根据这个顺序,
存储引擎可以确定不同的行数据给到mysql,因此mysql可以更新不同的行。存储引擎检查key的顺序不是确定性的。例如,InnoDB按照索引添加到
表的顺序检查键。

insert … on duplicate key 在执行时,innodb引擎会先判断插入的行是否产生重复key错误,如果存在,在对该现有的行加上S(共享锁)锁,如果返回该行数据给mysql,然后mysql执行完duplicate后的update操作,然后对该记录加上X(排他锁),最后进行update写入。
如果有两个事务并发的执行同样的语句,那么就会产生death lock

  • 6
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
可以使用MySQL的INSERT INTO ... ON DUPLICATE KEY UPDATE语句来实现存在修改不存在新增的操作。具体实现方法如下: 1. 使用唯一索引 首先,在MySQL中创建一个唯一索引,例如: CREATE UNIQUE INDEX idx_name ON table_name (column_name); 然后,使用以下语句进行操作: INSERT INTO table_name (column1, column2, ...) VALUES (value1, value2, ...) ON DUPLICATE KEY UPDATE column1=new_value1, column2=new_value2, ...; 其中,column1、column2等为表中的列名,value1、value2等为要插入的值,new_value1、new_value2等为要更新的值。 举例: CREATE TABLE user (id INT PRIMARY KEY, name VARCHAR(50) UNIQUE, age INT); INSERT INTO user (id, name, age) VALUES (1, 'Tom', 20) ON DUPLICATE KEY UPDATE age=21; 如果表中已经存在name为'Tom'的记录,则会将该记录的age更新为21;否则会插入一条新记录。 2. 使用REPLACE INTO语句 另一种实现方法是使用MySQL的REPLACE INTO语句,该语句会先尝试删除已有记录,再插入新记录。具体实现方法如下: REPLACE INTO table_name (column1, column2, ...) VALUES (value1, value2, ...); 举例: CREATE TABLE user (id INT PRIMARY KEY, name VARCHAR(50) UNIQUE, age INT); REPLACE INTO user (id, name, age) VALUES (1, 'Tom', 20); 如果表中已经存在name为'Tom'的记录,则会先删除该记录,再插入一条新记录;否则会直接插入一条新记录。 注意:使用REPLACE INTO语句会导致自增主键的值被重置,因此不建议在使用自增主键的表中使用该语句。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值