MySQL之存在则更新,否则就插入数据
背景
用户获得免费电影观看权,如果同个电影获得再次获得观看权,则在领取记录中更新获取时间,如果首次获取的免费电影观看权,则新增一条获奖记录。
思考
其实是数据库的“存在则更新数据,不存在就插入数据”问题,MySQL有相应的语法可以解决,需要搭配索引。
数据表结构
CREATE TABLE `prize_order` ( `id` int NOT NULL AUTO_INCREMENT, `mobile` char(11) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '手机号码', `order_status` int DEFAULT '0' COMMENT '受理状态,1-成功,0-失败', `create_time` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', `prize_id` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '奖品电影编号', PRIMARY KEY (`id`) USING BTREE, UNIQUE KEY `index_uni` (`mobile`,`prize_id`) USING BTREE ) ENGINE=InnoDB AUTO_INCREMENT=19 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci COMMENT='订单日志表';
索引
数据
第一种insert语法(推荐!!!)
语法(此时不知道表中已有prize_id='1'的数据)
insert into prize_order (mobile,create_time,prize_id) values('15122223333','2022-10-02 00:00:00','1') on DUPLICATE key update create_time =now()
不知道表中数据的情况下,已有数据会更新,mobile='15122223333' and prize_id='1'数据会修改create_time字段=now()
执行情况
语法(此时不知道表中无prize_id='60'的数据)
insert into prize_order (mobile,create_time,prize_id) values('15122223333',now(),'60') on DUPLICATE key update create_time = now()
不知道表中数据的情况下, 当数据不存在时,则会新增一条mobile='15122223333' ,prize_id='1',create_time字段=now() 的数据
执行情况
第二种replace语法(个别场景下使用)
语法(此时不知道表中已有prize_id='1'的数据)
replace into prize_order(mobile,order_status,create_time,prize_id)VALUES('15122223333','1',now(),'1')
执行情况
语法(此时不知道表中无prize_id='20'的数据)
replace into prize_order(mobile,order_status,create_time,prize_id)VALUES('15122223333','1',now(),'20')
执行结果
缺点
相信replace语法对很多“存在则更新,否则插入数据”的需求都满足。
我们也看到了,它会删除原数据,再进行插入数据,此时你有些字段不想更新,则被它抹除了,所以,用这个语法记得把所有需要的字段都正确赋值,不然会出现丢失数据的后果。
总结
根据以上对比,insert语法更优,推荐使用,因为不影响原有id,也只会更新所需字段,其余不处理的字段保持原有值;而replace语法则会改变id和改变除所需字段以外的值,有丢失数据的风险,建议熟悉该语法并对应业务场景才使用。
好了,以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
相关文章
MySQL与MSSQl使用While语句循环生成测试数据的代码
有时候我们测试性能的时候经常需要生产大量的测试数据,用sql语句直接生成的数据更快,需要的朋友可以参考下。2010-12-12
最新评论