mysql数据库删除重复数据只保留一条方法实例

作者:生有涯,知无涯 时间:2024-01-28 06:17:49 

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
    );

mysql数据库删除重复数据只保留一条方法实例

3.插入测试数据


insert into users values(0,'620616199409206512','张三'),
(0,'620616199409206512','张三'),
(0,'62062619930920651X','李四'),
(0,'62062619930920651X','李四'),
(0,'620622199101206211','王五'),
(0,'620622199101206211','王五'),
(0,'322235199909116233','赵六');

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

mysql数据库删除重复数据只保留一条方法实例

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数据库删除重复数据只保留一条方法实例

因为在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

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

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

mysql数据库删除重复数据只保留一条方法实例

继续更正:

给查询到的最大(或最小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了,如下图:

mysql数据库删除重复数据只保留一条方法实例

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
);

执行结果:

mysql数据库删除重复数据只保留一条方法实例

成功将重复的数据删除,只保留了最后一次增加的记录。同理也可以保留第一次添加的记录(即删除每个分组里面除最小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;

总结

来源:https://blog.csdn.net/qq_38923792/article/details/95240733

标签:mysql,删除,重复
0
投稿

猜你喜欢

  • 如何通过命令行进入python

    2022-10-28 22:12:56
  • 在SQL Server中处理空值时涉及的三个问题

    2009-02-05 15:30:00
  • java正则表达式验证工具类

    2022-03-03 12:47:01
  • python 中Arduino串口传输数据到电脑并保存至excel表格

    2022-05-17 05:41:11
  • element-ui表格列金额显示两位小数的方法

    2024-04-26 17:41:10
  • Python如何破解压缩包密码

    2023-09-01 19:15:41
  • Python 打印中文字符的三种方法

    2022-11-14 10:22:07
  • python爬虫可以爬什么

    2022-05-26 01:24:31
  • asp修改文件和文件夹的名字的代码

    2011-04-11 11:04:00
  • 解决pycharm安装scrapy DLL load failed:找不到指定的程序的问题

    2023-08-29 17:47:32
  • MySql nion与Limit查询介绍

    2024-01-23 15:53:52
  • python实现翻转棋游戏(othello)

    2022-06-02 10:40:19
  • MySQL 数据库范式化设计理论总结

    2024-01-12 17:29:07
  • ASP分段读取数据库代码

    2009-10-12 12:28:00
  • mysql导入导出数据中文乱码解决方法小结

    2024-01-19 17:47:45
  • SQL Server的链接服务器技术小结

    2024-01-17 03:41:22
  • Flash的Fallback Content等

    2010-04-01 12:18:00
  • 使用Python3 编写简单信用卡管理程序

    2021-01-17 20:57:59
  • Python 八个数据清洗实例代码详解

    2022-12-08 23:50:36
  • python字典的setdefault的巧妙用法

    2021-05-18 02:58:48
  • asp之家 网络编程 m.aspxhome.com