1.问题引入

假设一个场景,一张用户表,包含3个字段。id,identity_id,name。现在身份证号identity_id和姓名name有很多重复的数据,需要删除只保留一条有效数据。

2.模拟环境

  • 1.登入mysql数据库,创建一个单独的测试数据库mysql_exercise
create database mysql_exercise charset utf8;
  • 2.创建用户表users
create table users(id int auto_increment primary key,identity_id varchar(20),name varchar(20) not null);

  • 3.插入测试数据
insert into users values(0,'620616199409206512','张三'),(0,'620616199409206512','张三'),(0,'62062619930920651X','李四'),(0,'62062619930920651X','李四'),(0,'620622199101206211','王五'),(0,'620622199101206211','王五'),(0,'322235199909116233','赵六');

可以多执行几次,生成较多重复数据。

  • 4.解决思路
    (1)根据身份证号和name进行分组;
    (2)取出分组后的最大id(或最小id);
    (3)删除除最大(或最小)id以外的其他字段;

  • 5.第一次尝试(失败!!!)

delete from users where id not in (select max(id) from users group by identity_id,name);

报错:

1093 (HY000): You can't specify target table 'users' for update in FROM clause


因为在MYSQL里,不能先select一个表的记录,再按此条件进行更新和删除同一个表的记录。
解决办法是,将select得到的结果,再通过中间表select一遍,这样就规避了错误,
这个问题只出现于mysql,mssql和oracle不会出现此问题。

所以我们可以先将括号里面的sql语句先拿出来,先查到最大(或最小)id。

select max_id from (select max(id) as max_id from users group by identity_id,name);

接着,又报错了!!!

ERROR 1248 (42000): Every derived table must have its own alias

意思是说:提示说每一个衍生出来的表,必须要有自己的别名!

执行子查询的时候,外层查询会将内层的查询当做一张表来处理,所以我们需要给内层的查询加上别名

继续更正:
给查询到的最大(或最小id)结果当做一张新的表,起别名t,并查询t.mix_id

select t.max_id from (select max(id) as max_id from users group by identity_id,name) as t;

可以成功查到最大(或最小)id了,如下图:

  • 6.第二次尝试(成功!!!)
delete from users where id not in (select t.max_id from (select max(id) as max_id from users group by identity_id,name) as t);

执行结果:

成功将重复的数据删除,只保留了最后一次增加的记录。同理也可以保留第一次添加的记录(即删除每个分组里面除最小id以外的其他条记录)

3.知识拓展一:更新数据

其他场景应用:要将用户表user_info里名字(name)为空字符串("")的用户的状态(status)改成"0"

update user_info set status='0' where user_id in (select user_id from user_info where name='')

同样报了如下错误:

You can’t specify target table ‘user_info’ for update in FROM clause

因为在MYSQL里,不能先select一个表的记录,再按此条件进行更新和删除同一个表的记录,解决办法是,将select得到的结果,再通过中间表select一遍,这样就规避了错误。
以下两种均可!!!

update user_info set status='0' where user_id in (select user_id from (select user_id from user_info where name = '') t1);

下面这种也可,细微差别,别名可带as可不带,t1.user_id 直接和内层的user_id对应也可以。

update user_info set status='0' where user_id in (select t1.user_id from (select user_id from user_info where name='') as t1);

3.1 分步骤解析

(1)将以下查询结果作为中间表:

select user_id from user_info where name='';

(2)再查询一遍中间表作为结果集:

select user_id from (select user_id from user_info where name='') as t;

(3)更新数据

update user_info set status='0' where user_id in (select user_id from (select user_id from user_info where name='') as t1);

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  |
+----+------------------+
  • 解答一:
delete from Person where Id not in (select t.min_id from (select min(Id) as min_id from Person group by Email) as t);
  • 解答二:
delete p1 from Person as p1,Person as p2 where p1.Email=p2.Email and p1.Id > p2.Id;

mysql数据库删除重复的数据保留一条相关推荐

  1. mysql数据库删除重复的数据只保留一条

    问题引入 假设一个场景,一张用户表,包含 3 个字段:id,identity_id,name. 现在身份证号 identity_id 和姓名 name 有很多重复的数据,需要删除多余数据只保留一条有效 ...

  2. mysql删除重复的数据保留一条

    mysql删除重复的数据保留一条 -- 删除多余的重复记录,只保留最小id的记录,content 为内容重复字段 DELETE FROM zimis WHERE id IN (SELECT * FRO ...

  3. 删除重复的数据保留一条

    Mysql删除重复数据只保留一条_怪 咖@的博客-CSDN博客_mysql删除重复数据保留一条 这个是相关的链接,可以参考.

  4. sql删除重复的数据保留一条_leetcode题库-sql练习精讲系列--九、删除重复

    这是一个系列文章,这个系列的理念是通过一道题,搞懂一类题.涵盖了SQL面试最常考的知识点.搞懂这些题,面试时工作中sql不可能有问题. 文章分为引入问题-完整解析-答案-leetcode题和答案-知识 ...

  5. php去除重复的数据保留一条,mysql查找删除重复数据并只保留一条实例详解

    有这样一张表,表数据及结果如下: school_id school_name total_student test_takers 1239 Abraham Lincoln High School 55 ...

  6. MySql数据库去除重复的数据

    今天群内有群友提出了一个问题,就是MySql中删除重复数据的问题,然后回答了一下,发现正好接触到了之前可能没关注的一点儿小知识,在此做下简要记录(哦对,昨天晚上在试用腾讯云数据库的时候还遇到一个问题, ...

  7. 【数据库】Mysql删除重复记录只保留一条

    实际工作中,有可能会对数据库中的数据进行再次加工.假设有个表记录的是技术文章,有三个字段:st_link(文章链接).st_title(文章章节标题).st_name(文章名称) 假设初始时主键为st ...

  8. mysql类似于excel的删除重复项_sql删除重复项并保留其中一条(含sql优化)

    背景: 数据库包含重复数据,需要清理掉重复数据,并只保留其中一条. 结论 优化:百万数据查询删除重复数据,耗时从5423秒下降到2秒左右 优化过程: 根据搜索到的资料: 4.删除表中多余的重复记录(多 ...

  9. mysql删除重复记录只保留一条

    2019独角兽企业重金招聘Python工程师标准>>> 删除表中重复记录,只保留一条: delete from 表名 where 字段ID in (select * from (se ...

最新文章

  1. VC++ 进程间通信方法总结
  2. iframe中跨域页面访问parent的方法
  3. Mac OS使用技巧之十四:自定义文件图标
  4. 室内定位 - 资料收集
  5. 构造函数内部原理 包装类
  6. SQL Server 2014中的混合云和Hekaton功能
  7. Swift 类与结构体
  8. 傅里叶分析公式推导(最简单的傅里叶级数和傅里叶变换)
  9. Springboot打包部署到linux服务器的方法
  10. Android device monitor
  11. Java杂项基础知识点总结
  12. ArcMAP 添加注记与编辑
  13. 真杜比全景声家庭影院级投影设备,当贝做到了五千元内也支持
  14. python基础入门小结(1)
  15. 【RMF】ros机器人中间件框架学习系列一:了解原理
  16. TimescaleDB部署
  17. Python 类的定制
  18. 在conda虚拟环境中配置cuda+cudnn+pytorch深度学习环境(新手必看!简单可行!)
  19. Linux(Ubuntu)同步互联网时间(ntpdate)
  20. SMI-S 统一SAN管理

热门文章

  1. 一个对象实例化的过程(细节)
  2. 【Axure教程】分类筛选卡片(订单卡片案例)
  3. Unity中的UGUI源码解析之事件系统(6)-RayCaster(下)
  4. 微信公众号上传图片接口
  5. 自动驾驶中间件之四:主机厂自研
  6. CISSP复习笔记-第6章 通信与网络安全
  7. iOS自定义相机:带拍摄区域边框、半透明遮罩层、点击屏幕对焦、自动裁剪(含demo源码)
  8. 基于KNN算法的颜色识别
  9. 在线教育如何培养口碑效应?
  10. Java自动化测试(回写与断言 17)