SQL SERVER 实现相同记录为空显示(多列去除重复值,相同的只显示一条数据)
sql server语句查询中碰到结果集有重复数据,需要把这个重复数据汇总成一条显示。其余则正常显示。
使用SQL内置函数 ROW_NUMBER() 加 PARTITION 完成
ROW_NUMBER() OVER ( PARTITION BY '相同数据字段' ORDER BY GETDATE() ) row
PARTITION BY和GROUP BY类似。
GROUP BY会影响行数,针对于所有字段进行一个聚合。
PARTITION BY则不会影响行数,用做于此处刚刚好。
例:查询出字段有A、B、C、D、E。其中A代表姓名、B代表年龄、C代码性别、D代表爱好、E代表喜欢游戏。
其中A、B、C,这三个字段中有重复字段,而D、E则无重复字段。
那么语句便可以这样写
SELECT *,ROW_NUMBER() OVER ( PARTITION BY A,B,C ORDER BY GETDATE() ) row FROM tab
这样你便会看到你的结果中row字段会依据A、B、C三个字段依据不同值分别做一个分组。
| A | B | C | D | E | row |
| 小明 | 18 | 1 | 篮球 | LOL | 1 |
| 小明 | 18 | 1 | 乒乓球 | DNF | 2 |
| 小黄 | 18 | 2 | 游泳 | LOL | 1 |
| 小黄 | 18 | 2 | 滑板 | QQ飞车 | 2 |
| 小黄 | 18 | 2 | 跑步 | 战地五 | 3 |
| 小孙 | 18 | 1 | 睡觉 | 剑网三 | 1 |
所以以上数据便是依据PARTITION BY分组所显示行号。
既然已经得到这样的数据,那么把相同的数据更改为空就很简单了。使用CASE WHEN便可以实现。
SELECT CASE WHEN row = 1 THEN T.A
ELSE ''
END A ,
CASE WHEN row = 1 THEN T.B
ELSE ''
END B ,
CASE WHEN row = 1 THEN T.C
ELSE ''
END C ,
T.D ,
T.E
FROM ( SELECT * ,
ROW_NUMBER() OVER ( PARTITION BY A, B, C ORDER BY GETDATE() ) row
FROM tab
) T
那么以上语句执行最终结果便是一下表格里的数据:
| A | B | C | D | E | row |
| 小明 | 18 | 1 | 篮球 | LOL | 1 |
| 乒乓球 | DNF | 2 | |||
| 小黄 | 18 | 2 | 游泳 | LOL | 1 |
| 滑板 | QQ飞车 | 2 | |||
| 跑步 | 战地五 | 3 | |||
| 小孙 | 18 | 1 | 睡觉 | 剑网三 | 1 |
这样一来就简介多了,SQL SERVER很多内置函数还是非常好用的。
最后放上一个测试临时表格供大家测试吧!
CREATE TABLE #tab
(
headerNo VARCHAR(10) ,
machineNO VARCHAR(10) ,
descrption NVARCHAR(20) ,
artNo VARCHAR(20) ,
qty INT ,
repartno VARCHAR(20) ,
repqty INT
)
插入数据
insert INTO #tab SELECT 'HD01','0101520',N'电池出问题','102020',2,'102020',2
insert INTO #tab SELECT 'HD01','0101520',N'电池出问题','101010',2,'202020',2
insert INTO #tab SELECT 'HD01','0101520',N'电池出问题','126888',2,'102020',2
insert INTO #tab SELECT 'HD02','01012221',N'D电机故障','102020',2,'102020',2
insert INTO #tab SELECT 'HD03','12312312',N'突然停机','102020',2,'102020',2
insert INTO #tab SELECT 'HD03','12312312',N'突然停机','102020',2,'102020',2
insert INTO #tab SELECT 'HD04','12312344',N'皮带松了','102020',2,'102020',2
语句查询执行
SELECT CASE WHEN row = 1 THEN headerNo
ELSE ''
END headerNo ,
CASE WHEN row = 1 THEN machineNO
ELSE ''
END machineNO ,
CASE WHEN row = 1 THEN descrption
ELSE ''
END descrption ,
artNo ,
qty ,
repartno ,
repqty
FROM ( SELECT * ,
ROW_NUMBER() OVER ( PARTITION BY headerNo, machineNO,
descrption ORDER BY GETDATE() ) row
FROM #tab
) M
以上测试SQL语句来自博客园以下老哥的文章,它的文章只编写了SQL语句供予测试。
我加以修改,做了一些理解性的分析。
https://www.cnblogs.com/panxuguang/p/5668520.html
后续更新
PARTITION BY简称分区函数,GROUP BY为聚合函数。
PARTITION BY的使用之处可以有很多,文章的上半部分查询出来的数据用于报表非常合适。
但用作于程序中就会显得比较复杂。
以此可以思考,既然 PARTITION BY是依据当前字段不同的字段做一个分组,不影响数据结构、数据行。
结合ROW_NUMBER对分组后的数据进行一个排序。这样就可以根据ROU_NUMBER后的字段进行数据查询。
同时,PARTITION BY 还可以在一个查询语句中针对于多个字段进行查询。但后续的PARTITION BY须携带上前一个字段。
比如上面例子中再增加一个字段,父母标识。L字段中,0代表自己,1代表母亲,2代表父亲。D和E就表示父母的爱好。
SELECT CASE WHEN row = 1 THEN T.A
ELSE ''
END A ,
CASE WHEN row = 1 THEN T.B
ELSE ''
END B ,
CASE WHEN row = 1 THEN T.C
ELSE ''
END C ,
CASE WHEN row2 = 1 THEN T.L
ELSE ''
END F,
T.D ,
T.E
FROM ( SELECT * ,
ROW_NUMBER() OVER ( PARTITION BY A, B, C ORDER BY GETDATE() ) row,
ROW_NUMBER() OVER ( PARTITION BY A, B, C, L ORDER BY GETDATE() ) row2
FROM tab
) T
这样除了第一次的依据A,B,C进行了一个内部分区排序,
随后第二次操作就会依据A,B,C为参照针对L的不同字段在做一个内部分区排序。数据就是下面的样式
| A | B | C | L | D | E |
| 小明 | 18 | 1 | 0 | 篮球 | LOL |
| 乒乓球 | DNF | ||||
| 2 | 篮球 | 吃鸡 | |||
| 跳伞 | 黎明杀机 | ||||
| 1 | 跳绳 | 和平精英 | |||
| 小黄 | 18 | 2 | 游泳 | LOL | |
| 滑板 | QQ飞车 | ||||
| 跑步 | 战地五 | ||||
| 小孙 | 18 | 1 | 睡觉 | 剑网三 |
通过PARTITON BY 分区排序之后,你还可以取以什么为条件了前三条,以什么字段为排序的第一条等等操作
SQL SERVER 实现相同记录为空显示(多列去除重复值,相同的只显示一条数据)的更多相关文章
- SQL Server 最小日志记录
SQL Server之所以记录事务日志,首要目的是为了把失败或取消的操作还原到最原始的状态,但是,并不是所有的操作都需要完全记录事务日志,比如,在一个空表上放置排他锁,把大量的数据插入到该空表中.即使 ...
- SQL Server 查询表的记录数(3种方法,推荐第一种)
http://blog.csdn.net/smahorse/article/details/8156483 --SQL Server 查询表的记录数 --one: 使用系统表. SELECT obje ...
- 【转】SQL Server 查询表的记录数(3种方法,推荐第一种)
--SQL Server 查询表的记录数 --one: 使用系统表. SELECT object_name (i.id) TableName, rows as RowCnt FROM sysindex ...
- sql server显示某一列中有重复值的行
sql server查询一张表 ,显示某一列中有重复值的行,可以这样写: Select * From 表名 where 列名 in(Select 列名 From Table group by 列名 h ...
- SQL SERVER 判断是否存在数据库、表、列、视图
SQL SERVER 判断是否存在数据库.表.列.视图 --1. 判断数据库是否存在 IF EXISTS (SELECT * FROM SYS.DATABASES WHERE NAME = '数据库名 ...
- img只显示图片一部分 或 css设置背景图片只显示图片指定区域
17:14 2016/3/22img只显示图片一部分 或 css设置背景图片只显示图片指定区域 background-position: 100% 56%; 设置背景图片显示图片的哪个坐标区域,图片左 ...
- sql server相邻表记录交换(单双两两交换)
在博客园的博问中看到了一个这样的提问:一个表中有id和name两个字段,id是连续非空不重复的,想要交换相邻表记录的name值(单双两两交换). 另外,如果最后的单独行没有对应的下一行匹配记录的话,就 ...
- SQL SERVER常用语法记录
用于记录SQL SERVER常用语法,以及内置函数. 以下语句包含: WITH 临时表语法 ROW_NUMBER()内置函数,我一般主要是用来分页.针对于查出来的所有数据做一个数字排序 分页的BETW ...
- SQL Server 2005中的分区表(二):如何添加、查询、修改分区表中的数据(转)
在创建完分区表后,可以向分区表中直接插入数据,而不用去管它这些数据放在哪个物理上的数据表中.接上篇文章,我们在创建好的分区表中插入几条数据 insert Sale ([Name],[SaleTime] ...
随机推荐
- ThnikPHP3.2 学习链接整理
ThnikPHP3.2 学习链接整理 ThinkPHP3.2.3 U()方法的使用总结 看云手册 ThinkPHP3.2完全开发手册 TP3.2单字母函数 TP3.x中 M方法和D方法的区别
- 服务器安装node全教程
我的服务器centos,安装node时出了点小麻烦,在这里记述我的方法. 1.进入node下载网站https://nodejs.org/en/download/,这里右键复制下载链接 2.进入cent ...
- chrom里面的performance 颜色
在network里面,在network里面,在network里面(重要事件说三遍) : 1. HTML 文件为蓝色. 2. 脚本为黄色. 3. 样式表为紫色. 4. 媒体文件为绿色. 5. 其他资源为 ...
- PHP正则匹配各种匹配方法
平时做网站经常要用正则表达式,下面是一些讲解和例子,仅供大家参考和修改使用: 匹配数字 "^\d+$" //非负整数(正整数 + 0) "[1][1-9][0-9]$&q ...
- Android内核模块编译执行
Author: GeneBlue 0X01 前言 内核驱动是漏洞的高发区,了解Android驱动代码的编写是分析.利用驱动漏洞的基础.本文以一个"hello"驱动为例,简单介绍内核 ...
- Linux配置NTP时间服务器(date、hwclock、NTP服务器的配置)
目录 date命令 hwclock命令 NTP服务的部署 服务端 客户端 date命令 date 命令的作用是查看和设置Linux中的系统日期时间 date ...
- php、jsp、asp和aspx的区别
目录 PHP JSP ASP ASP.NET PHP PHP是一种跨平台的服务器端的嵌入式脚本语言.它大量地借用C.Java 和 Perl 语言的语法,并耦合PHP自己的特性,使WEB开发者能够快速地 ...
- CVE-2017-11826:Office Open XML 标签嵌套解析混淆漏洞
\x01 前言 CVE-2017-11826 据说是 360 在 2017 年 9 月底发现的一个关于 XML 格式解析的一个漏洞,之后微软在 10 月份发布了关于 CVE-2017-11826 的补 ...
- Intel汇编语言程序设计学习-第四章 数据传送、寻址和算术运算-下
4.3 和数据相关的操作符和伪指令 操作符和伪指令并非机器可执行的指令,相反,它们是由汇编器进行解释的.开发者可以使用一系列的MASM操作符或伪指令获取数据的地址以及大小等特征信息: OFFSET操 ...
- web技术培训(一)-云服务器、域名相关
云服务器 什么是云服务器(这部分可以跳过) 云服务器(Elastic Compute Service, ECS)是一种简单高效.安全可靠.处理能力可弹性伸缩的计算服务.其管理方式比物理服务器更简单高效 ...