mysql数据库删除重复数据只保留一条方法实例
1.问题引入
假设一个场景,一张用户表,包含3个字段。id,identity_id,name。现在身份证号identity_id和姓名name有很多重复的数据,需要删除只保留一条有效数据。
2.模拟环境
1.登入mysql数据库,创建一个单独的测试数据库mysql_exercise
createdatabasemysql_exercisecharsetutf8;
2.创建用户表users
createtableusers( idintauto_incrementprimarykey, identity_idvarchar(20), namevarchar(20)notnull );
3.插入测试数据
insertintousersvalues(0,'620616199409206512','张三'), (0,'620616199409206512','张三'), (0,'62062619930920651X','李四'), (0,'62062619930920651X','李四'), (0,'620622199101206211','王五'), (0,'620622199101206211','王五'), (0,'322235199909116233','赵六');
可以多执行几次,生成较多重复数据。
4.解决思路
(1)根据身份证号和name进行分组;
(2)取出分组后的最大id(或最小id);
(3)删除除最大(或最小)id以外的其他字段;
5.第一次尝试(失败!!!)
deletefromuserswhereidnotin(selectmax(id)fromusersgroupbyidentity_id,name);
报错:
1093(HY000):Youcan'tspecifytargettable'users'forupdateinFROMclause
所以我们可以先将括号里面的sql语句先拿出来,先查到最大(或最小)id。
selectmax_idfrom(selectmax(id)asmax_idfromusersgroupbyidentity_id,name);
接着,又报错了!!!
ERROR1248(42000):Everyderivedtablemusthaveitsownalias
意思是说:提示说每一个衍生出来的表,必须要有自己的别名!
执行子查询的时候,外层查询会将内层的查询当做一张表来处理,所以我们需要给内层的查询加上别名
继续更正:
给查询到的最大(或最小id)结果当做一张新的表,起别名t,并查询t.mix_id。
selectt.max_idfrom(selectmax(id)asmax_idfromusersgroupbyidentity_id,name)ast;
可以成功查到最大(或最小)id了,如下图:
6.第二次尝试(成功!!!)
deletefromuserswhereidnotin( selectt.max_idfrom (selectmax(id)asmax_idfromusersgroupbyidentity_id,name)ast );
执行结果:
成功将重复的数据删除,只保留了最后一次增加的记录。同理也可以保留第一次添加的记录(即删除每个分组里面除最小id以外的其他条记录)
3.知识拓展一:更新数据
其他场景应用:要将用户表user_info里名字(name)为空字符串("")的用户的状态(status)改成"0"
updateuser_infosetstatus='0'whereuser_idin(selectuser_idfromuser_infowherename='')
同样报了如下错误:
Youcan'tspecifytargettable‘user_info'forupdateinFROMclause
因为在MYSQL里,不能先select一个表的记录,再按此条件进行更新和删除同一个表的记录,解决办法是,将select得到的结果,再通过中间表select一遍,这样就规避了错误。
以下两种均可!!!
updateuser_infosetstatus='0'whereuser_idin (selectuser_idfrom(selectuser_idfromuser_infowherename='')t1);
下面这种也可,细微差别,别名可带as可不带,t1.user_id直接和内层的user_id对应也可以。
updateuser_infosetstatus='0'whereuser_idin (selectt1.user_idfrom(selectuser_idfromuser_infowherename='')ast1);
3.1分步骤解析
(1)将以下查询结果作为中间表:
selectuser_idfromuser_infowherename='';
(2)再查询一遍中间表作为结果集:
selectuser_idfrom(selectuser_idfromuser_infowherename='')ast;
(3)更新数据
updateuser_infosetstatus='0'whereuser_idin (selectuser_idfrom(selectuser_idfromuser_infowherename='')ast1);
4.拓展练习:删除重复数据
编写一个SQL查询,来删除Person表中所有重复的电子邮箱,重复的邮箱里只保留Id最小的那个。
+----+------------------+ |Id|Email| +----+------------------+ |1|john@example.com| |2|bob@example.com| |3|john@example.com| +----+------------------+
Id是这个表的主键。
例如,在运行你的查询语句之后,上面的Person表应返回以下几行:
+----+------------------+ |Id|Email| +----+------------------+ |1|john@example.com| |2|bob@example.com| +----+------------------+
解答一:
deletefromPersonwhereIdnotin( selectt.min_idfrom( selectmin(Id)asmin_idfromPersongroupbyEmail )ast );
解答二:
deletep1from Personasp1,Personasp2 wherep1.Email=p2.Emailandp1.Id>p2.Id;
总结
到此这篇关于mysql数据库删除重复数据的方法只保留一条的文章就介绍到这了,更多相关mysql删除重复数据内容请搜索毛票票以前的文章或继续浏览下面的相关文章希望大家以后多多支持毛票票!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。