选择重复,消除重复和选择出序列 

有例表:emp 

emp_no   name    age     
    001            Tom      17      
    002            Sun       14      
    003            Tom       15      
    004            Tom      16  

要求: 

列出所有名字重复的人的记录 

( 1 )最直观的思路:要知道所有名字有重复人资料,首先必须知道哪个名字重复了: 

select    name    from    emp        group     by    name      having     count ( * ) > 1  


 所有名字重复人的记录是: 

select     *     from    emp 
    where  name    in    ( select    name    from    emp  group     by    name  having   count ( * ) > 1 ) 

( 2 )稍微再聪明一点,就会想到,如果对每个名字都和原表进行比较,大于2个人名字与这条记录相同的就是合格的 ,就有 

select     *     from    emp    where    ( select     count ( * )    from    emp   e     where    e.name = emp.name)    > 1  

-- 注意一下这个>1,想下如果是 =1,如果是 =2 如果是>2 如果 e 是另外一张表 而且是=0那结果 就更好玩了:) 

这个过程是 在判断工号为001的 人 的时候先取得 001的 名字(emp.name) 然后和原表的名字进行比较 e.name 

注意e是emp的一个别名。 

再稍微想得多一点,就会想到,如果有另外一个名字相同的人工号不与她他相同那么这条记录符合要求: 

select     *     from    emp     
     where     exists      
                  ( select     *     from    emp   e     where    e.name = emp.name    and    e.emp_no <> emp.emp_no) 

 此思路的join写法: 

select    emp. *         from    emp,emp e
        where emp.name = e.name  and  emp.emp_no <> e.emp_no /**/
/*     这个语句较规范的   join   写法是     
select emp.* from   emp   inner join emp   e     on emp.name=e.name and emp.emp_no<>e.emp_no     
但个人比较倾向于前一种写法,关键是更清晰     */      
b、有例表: emp     
name     age     
Tom        16      
Sun        14      
Tom       16      
Tom       16  

-- --------------------------------------------------清除重复----------------------------------------------------
过滤掉所有多余的重复记录 
( 1 )我们知道distinct、 group   by  可以过滤重复,于是就有最直观的 
 
select     distinct     *     from    emp     或      select    name,age    from    emp    group     by    name,age 

获得需要的数据,如果可以使用临时表就有解法: 
 
select     distinct     *     into    #tmp     from    emp   
    delete     from    emp   
    insert     into    emp    select     *     from    #tmp 

( 2 )但是如果不可以使用临时表,那该怎么办? 
我们观察到我们没办法区分数据(物理位置不一样,对 SQL Server来说没有任何区别),思路自然是想办法把数据区分出来了,既然现在的所有的列都没办法区分数据,唯一的办法就是再加个列让它区分出来,加什么列好?最佳选择是identity列: 
 
alter     table    emp    add    chk    int     identity ( 1 , 1 ) 

 表示例: 
 
name   age   chk     
    Tom     16      1      
    Sun      14      2      
    Tom     16      3      
    Tom     16      4  

重复记录可以表示为: 
 
select     *     from    emp  where  ( select     count ( * )    from    emp   e    where    e.name = emp.name) > 1  

 要删除的是: 
 
delete     from    emp 
    where  ( select     count ( * )    from    emp   e      where    e.name = emp.name    and    e.chk >= emp.chk) > 1  
 
再把添加的列删掉,出现结果。 
 
alter     table    emp    drop     column    chk 

 
( 3 )另一个思路: 
视图 
 
select     min (chk)  from    emp  group     by    name  having     count ( * )    > 1  

 获得有重复的记录chk最小的值,于是可以 
 
delete   from    emp  where  chk    not     in  ( select   min (chk)  from    emp  group     by    name) 

写成join的形式也可以: 
 
( 1 )有例表:emp 
 
emp_no    name    age     
    001             Tom      17      
    002             Sun       14      
    003             Tom      15      
    004             Tom      16  

 ◆要求生成序列号 
( 1 )最简单的方法,根据b问题的解法: 
 
alter     table    emp    add    chk    int     identity ( 1 , 1 )   或   
    select     * , identity ( int , 1 , 1 )   chk    into    #tmp    from    emp 

 ◆如果需要控制顺序怎么办? 
 
select     top     100000     * , identity ( int , 1 , 1 )   chk    into    #tmp    from    emp    order     by    age 

 ( 2 ) 假如不可以更改表结构,怎么办? 
如果不可以唯一区分每条记录是没有办法的,在可以唯一区分每条记录的时候,可以使用a 中的count的思路解决这个问题 
 
select    emp. * ,( select     count ( * )    from    emp   e    where    e.emp_no <= emp.emp_no)   
    from    emp   
    order     by    ( select     count ( * )    from    emp   e    where    e.emp_no <= emp.emp_no) 

SQL查询重复数据和清除重复数据相关推荐

  1. SQL查询从多个表返回数据

    本文翻译自:SQL query return data from multiple tables I would like to know the following: 我想知道以下内容: how t ...

  2. SQL查询最大值,返回整行数据

    SQL查询最大值,返回整行数据 1.问题 部分数据如下,未完整展示.如何从 table_a 表中查询 p_postions 列的最大值对应的一行数据? 2.解答 方法1:先排序,再取第一条 SELEC ...

  3. SQL 查询字段包含特殊符号的数据

    SQL 查询字段包含特殊符号的数据 SQL 查询字段中包含单引号 查询的时候输入两个单引号就转义了 ,chr(39) 表示单引号.chr(40)表示双引号 eg: select * from XX.T ...

  4. SQL查询两个表不同的数据

    查询两个表不同的数据 select * from B where (select count(1) from A where A.ID = B.ID) = 0 查询两个表相同的数据可以用union a ...

  5. sql查询远程数据库的表的数据并填充到本地数据库的表

    insert into A select A.ID,A.NAME from opendatasource('SQLOLEDB','Password=123;Persist Security Info= ...

  6. mysql查询结果随机取一个_从SQL查询结果随机取一条数据

    最近工作中会涉及复杂的SQL语句,还是挺有意思的.以前想写没有应用场景,现在正好可以多锻炼锻炼.我觉得在实际开发中,程序代码并不是最难的,最难的还是SQL.SQL逻辑性最强,而且还要很清楚表设计,业务 ...

  7. 一条sql查询多个表的统计数据

    环境:Mysql+Mybatis 需求:每个省份单独有一张表,每个表统计"待提交"."待审核"."已通过"和"总条数"信 ...

  8. sql 查询之排除某种特定的数据

    -- 原始数据 SELECT 'LAB TEST' A,'RULE1' B,'TEST MSG' C,'TRUE' D UNION ALL SELECT 'LAB TEST','RULE2','TES ...

  9. SQL查询获取最后一笔时间数据问题

    数据库表如下: 列A    列B  列C AA     20    2011-10-17 00:00:00 BB     20    2011-10-17 00:00:00 AA     20    ...

最新文章

  1. 2021年大数据Hadoop(四):Hadoop发行版公司
  2. java中velocity定义宏标签_velocity自定义标签和指令(转:zwj)
  3. 胆固醇竟是新冠最隐蔽同伙!协助病毒入侵细胞,普林斯顿、哈佛医学院首次观察到全过程...
  4. rtmp 推送h264 + aac 的数据
  5. linux centos7.6.1810安装nfs命令整理
  6. confirm弹框修改按钮确认取消为是否
  7. python 捕获鼠标点击事件,在Python中的wx.Frame外部捕获鼠标事件
  8. python经纬度获取县名_利用 Python 批量获取县镇运输距离
  9. MYSQL数据库表中字段追加字符串内容
  10. python画龙猫_微信小程序支付demo,后端使用python
  11. layui实现简单分页
  12. 螃蟹效应:表现优秀遭嫉妒,如何应对职场竞争?
  13. python 爬取直播弹幕视频_Python实现抓取斗鱼实时弹幕
  14. WPF图片背景色过滤
  15. 汽车驱动桥毕业设计【毕业设计(论文) +7张CAD图纸】
  16. 解析Unity3D中计算法线矩阵的函数
  17. 我的七条人生哲理以及个人学习方法总结
  18. cad展点kszd小程序_cad实现批量展坐标点方法介绍
  19. VBA批量OCR识别提取身份证照片信息_白描网页版 - 高效准确且免费的OCR文字识别工具...
  20. 揭秘拳头公司的游戏API: 充分发挥ZUUL的性能

热门文章

  1. MS SQL存储过程
  2. signaturenonce php,php微信公众平台开发--接入微信API
  3. Lichee(二) 在sun4i_crane平台下的编译
  4. soloV2保姆级教程(含环境配置,训练自己的数据集,代码逻辑分析等。能踩得坑都踩了....)更新ing
  5. 技术干货|昇思MindSpore 1.5版本中的亲和算法库——MindSpore Boost
  6. EAS 取数公式二次开发指南
  7. 如何查询Office版本号
  8. windows7图形界面远程centos6.5
  9. c++ 数组作为参数、返回值
  10. 关于SQL注入靶场搭建及过关教程