ROW_NUMBER、RANK、DENSE_RANK的用法 (加补充)

 

ROW_NUMBER RANK DENSE_RANK的用法

 

  (爱新觉罗 .毓华   2007-12-16  广东深圳 )
  SQL Server 2005 引入几个新的排序 (排名 )函数 , ROW_NUMBER RANK DENSE_RANK等。 这些新函数使您可以有效地分析数据以及向查询的结果行提供排序值。  
  --------------------------------------------------------------------------
  ROW_NUMBER()

  说明:返回结果集分区内行的序列号,每个分区的第一行从 1 开始。
  语法: ROW_NUMBER () OVER ( [  <partition_by_clause> ]  <order_by_clause> )


  备注: ORDER BY 子句可确定在特定分区中为行分配唯一 ROW_NUMBER 的顺序。


  参数:

<partition_by_clause> :将 FROM 子句生成的结果集划入应用了 ROW_NUMBER 函数的分区。  
  <order_by_clause>:确定将 ROW_NUMBER 值分配给分区中的行的顺序。
 

  返回类型: bigint
 

简单实例:

CREATE TABLE testdd

    ( deptno                         NUMBER (*, 0 ),

    ename                          VARCHAR2 ( 4000 ),

    sal                            NUMBER ,

    team                           VARCHAR2 ( 4000 ))

/

 

1.以 deptno , team 分区(组),按照 sal 排序

select deptno , team , row_number () over ( partition by deptno , team order by sal desc ) rn , ename , sal

from testdd

 

DEPTNO

TEAM

RN

ENAME

SAL

1

101

1

Lim

4000

1

101

2

Lee

3000

1

101

3

Qin

2500

1

102

1

Xu

50000

1

102

2

Wang

3100

1

102

3

Zhu

2400

2

102

1

Wong1

2200

2

102

2

Wong2

2000

2

103

1

Ding

3500

2

103

2

Ding1

2500

2

103

3

Ding2

2200

2

103

4

Ding3

2000

 

 

 

 

 

 

 

 

 

2.整个 table按照 sal 排序

select deptno , team , row_number () over ( order by sal desc ) rn , ename , sal

from testdd

 

DEPTNO

TEAM

RN

ENAME

SAL

1

102

1

Xu

50000

1

101

2

Lim

4000

2

103

3

Ding

3500

1

102

4

Wang

3100

1

101

5

Lee

3000

1

101

6

Qin

2500

2

103

7

Ding1

2500

1

102

8

Zhu

2400

2

102

9

Wong1

2200

2

103

10

Ding2

2200

2

103

11

Ding3

2000

2

102

12

Wong2

2000

 

 

3.以 deptno 分区(组),按照 sal 排序

select deptno , team , row_number () over ( partition by deptno order by sal desc ) rn , ename , sal

from testdd

 

DEPTNO

TEAM

RN

ENAME

SAL

1

102

1

Xu

50000

1

101

2

Lim

4000

1

102

3

Wang

3100

1

101

4

Lee

3000

1

101

5

Qin

2500

1

102

6

Zhu

2400

2

103

1

Ding

3500

2

103

2

Ding1

2500

2

102

3

Wong1

2200

2

103

4

Ding2

2200

2

102

5

Wong2

2000

2

103

6

Ding3

2000

 

4.使用实例: (去掉重复记录 )

 

update INQUIRE_ALL_DETAIL_CURRENT

set email_type = 'P'

--delete from INQUIRE_ALL_DETAIL_CURRENT

--delete from inquire_all_email_current

--delete from INQUIRE_ALL_EMAIL_ADDRESS_cur

  where email_id in

(

select distinct ( email_id )

FROM ( SELECT INQUIRE_ALL_DETAIL_CURRENT .*, ROW_NUMBER () OVER ( PARTITION BY INQUIRY_ID , supplier_id , product_id , email_type ORDER BY EMAIL_ID DESC ) RN ,

      COUNT ( 1 ) OVER ( PARTITION BY INQUIRY_ID , supplier_id , product_id , email_type ) COUNTS

  FROM INQUIRE_ALL_DETAIL_CURRENT ) A

  WHERE COUNTS = 2 and rn = 1

  )

 

5.利用 CTE来过滤 ROW_NUMBER()的用法 (选择出每个区的前三名):

 

--CTE filter row_number

  with NumberedRows AS

  (

  select deptno , team , row_number () over ( partition by deptno order by sal desc ) rn , ename , sal

  from testdd

)

  select * from NumberedRows

  where rn between 1 and 3



DEPTNO

TEAM

RN

ENAME

SAL

1

101

1

Lim

50000

1

102

2

Xu

50000

1

102

3

Wang

3100

2

103

1

Ding

3500

2

103

2

Ding1

2500

2

102

3

Wong1

2200

 

  --------------------------------------------------------------
  RANK()

  说明:返回结果集的分区内每行的排名。行的排名是相关行之前的排名数加一。
  语法: RANK () OVER ( [  < partition_by_clause > ]  < order_by_clause > )
  备注:如果两个或多个行与一个排名关联,则每个关联行将得到相同的排名。
        例如,如果两位顶尖销售员具有同样的 SalesYTD 值,他们将并列第一。
        由于已有两行排名在前,所以具有下一个最大 SalesYTD 的销售人员将排名第三。
        因此, RANK 函数并不总返回连续整数。  
        用于整个查询的排序顺序决定了行在结果集中的显示顺序。这也隐含了行在每个分区中的排名。
  参数: < partition_by_clause > :将 FROM 子句生成的结果集划分为要应用 RANK 函数的分区。  
         < order_by_clause >:确定将 RANK 值应用于分区中的行时所基于的顺序。
  返回类型: bigint

 

简单例子:

1.按 deptno 分区 排名

SELECT  RANK () OVER ( PARTITION BY deptno order by sal desc ) as RANK , deptno , team , ename , sal

  FROM testdd 

 

1

1

101

Lim

50000

1

1

102

Xu

50000

3

1

102

Wang

3100

4

1

101

Lee

3000

5

1

101

Qin

2500

6

1

102

Zhu

2400

1

2

103

Ding

3500

2

2

103

Ding1

2500

3

2

102

Wong1

2200

3

2

103

Ding2

2200

5

2

102

Wong2

2000

5

2

103

Ding3

2000

 

2.按每个 depno 内的 team 分区排名

SELECT  RANK () OVER ( PARTITION BY deptno , team order by sal desc ) as RANK , deptno , team , ename , sal FROM testdd 

 

RANK

DEPTNO

TEAM

ENAME

SAL

1

1

101

Lim

50000

2

1

101

Lee

3000

3

1

101

Qin

2500

1

1

102

Xu

50000

2

1

102

Wang

3100

3

1

102

Zhu

2400

1

2

102

Wong1

2200

2

2

102

Wong2

2000

1

2

103

Ding

3500

2

2

103

Ding1

2500

3

2

103

Ding2

2200

4

2

103

Ding3

2000


 
3.按每个 depno 内的平均分以及排名

    -- get average number

  with teamTotal AS

  (

  select deptno , team , sum ( sal ) as sumSal from testdd group by deptno , team

  )

 

  select deptno , team , sumSal ,

  rank () OVER ( PARTITION BY deptno ORDER BY sumSal , deptno DESC ) AS Rank ,

  AVG ( sumSal ) OVER ( PARTITION BY deptno ) AS Average

  from teamTotal

 

DEPTNO

TEAM

SUMSAL

RANK

AVERAGE

1

101

55500

1

55500

1

102

55500

1

55500

2

102

4200

1

7200

2

103

10200

2

7200

 

 

DENSE_RANK()
 
  说明:返回结果集分区中行的排名,在排名中没有任何间断。行的排名等于所讨论行之前的所有排名数加一
  语法: DENSE_RANK () OVER ( [  < partition_by_clause > ]  < order_by_clause > )
  备注:如果有两个或多个行受同一个分区中排名的约束,则每个约束行将接收相同的排名。
        例如,如果两位顶尖销售员具有相同的 SalesYTD 值,则他们将并列第一。
        接下来 SalesYTD 最高的销售人员排名第二。该排名等于该行之前的所有行数加一。
        因此, DENSE_RANK 函数返回的数字没有间断,并且始终具有连续的排名。  
        整个查询所用的排序顺序确定了各行在结果中的显示顺序。这说明排名第一的行可以不是分区中的第一行

  参数: < partition_by_clause > :将 FROM 子句所生成的结果集划分为数个将应用 DENSE_RANK 函数的分区。  
         < order_by_clause >:确定将 DENSE_RANK 值应用于分区中各行的顺序。
  返回类型: bigint

简单实例:

1.按 deptno 分区 排名

SELECT  dense_RANK () OVER ( PARTITION BY deptno order by sal desc ) as RANK , deptno , team , ename , sal FROM testdd 

 

RANK

DEPTNO

TEAM

ENAME

SAL

1

1

101

Lim

50000

1

1

102

Xu

50000

2

1

102

Wang

3100

3

1

101

Lee

3000

4

1

101

Qin

2500

5

1

102

Zhu

2400

1

2

103

Ding

3500

2

2

103

Ding1

2500

3

2

102

Wong1

2200

3

2

103

Ding2

2200

4

2

102

Wong2

2000

4

2

103

Ding3

2000

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值