你真的会玩 SQL 吗?冷落的 Top 和 Apply

转载 2015年11月20日 09:19:43

本文地址:http://blog.csdn.net/shanglianlm/article/details/49942869

本章预先想写一些Top和Apply基本的用法,但好像没什么意义,所以删掉了一些无用的东西,只留下几个示例,以保证系列的完整性。

Top和Apply解决的常见问题,如返回每个雇员的3个最新订单,订单的时间越新优先级就越高,但还需要引入一个决胜属性,以确定时间桢的订单的优先级,如可用id作为决胜属性。这里提供的解决方案比其它方案要简单得多,且执行速度更快。

返回每个雇员的3个最新订单:

SELECT  empid ,
        orderid ,
        custid ,
        orderdate ,
        requireddate
FROM    sales.orders AS o1
WHERE   orderid IN ( SELECT TOP 3
                            orderid
                     FROM   sales.orders AS o2
                     WHERE  o2.empid = o1.empid
                     ORDER BY orderdate DESC ,
                            orderid DESC )

运用APPLY解决:

SELECT  e.empid ,
        a.orderid ,
        a.custid ,
        a.orderdate ,
        a.requireddate
FROM    hr.employees AS e
        CROSS APPLY ( SELECT TOP 3
                                orderid ,
                                custid ,
                                orderdate ,
                                requireddate
                      FROM      sales.orders AS o
                      WHERE     o.empid = e.empid
                      ORDER BY  orderdate DESC ,
                                orderid DESC
                    ) AS a

先扫描employees 获得empid,对每个empid值对orders表查询返回 该雇员的3个最新订单。这里可以返回多个属性。

还有一种解决方案在特定情况下竟然比使用APPLY运算符的方法还要快,使用ROW_NUMBER函数。先为每个订单计算行号,按empid进行分区,并按orderdate desc, orderid desc 顺序排序。然后在外部查询中,只筛选行号小于或等于3的行。

如下:

SELECT  orderid ,
        custid ,
        orderdate ,
        requireddate
FROM    ( SELECT    orderid ,
                    custid ,
                    orderdate ,
                    requireddate ,
                    ROW_NUMBER() OVER ( PARTITION BY empid ORDER BY orderdate DESC , orderid DESC ) AS rownum
          FROM      sales.orders
        ) AS d
WHERE   rownum  3

练习:

从学生表中选取对应班级的前num名学生成绩

--显示结果
/*
bj         xh   name       cj
---------- ---- ---------- -----------
一班         A006 A6         100
一班         A005 A5         99
一班         A001 A1         89
一班         A002 A2         89

二班         B001 B7         100
二班         B001 B6         99
二班         B001 B9         97
二班         B001 B8         90
二班         B001 B5         88
*/
-- 创建测试表
declare @student table(        
---学生表
bj varchar(10),          
-- 班级
xh char(4),                 
-- 学号
name varchar(10),         
-- 姓名   
cj int)                     
-- 成绩

declare @tj table(            
---统计表
bj varchar(10),          
-- 班级
num int)                 
-- 人数  :从学生表中选取对应班级的前num名学生成绩 

set nocount on

-- 添加测试数据
insert @student select '一班' ,'A001','A1',89
insert @student select '一班' ,'A002','A2',89
insert @student select '一班' ,'A003','A3',59
insert @student select '一班' ,'A004','A4',80
insert @student select '一班' ,'A005','A5',99
insert @student select '一班' ,'A006','A6',100
insert @student select '一班' ,'A007','A7',82

insert @student select '二班' ,'B001','B1',19
insert @student select '二班' ,'B001','B2',81
insert @student select '二班' ,'B001','B3',69
insert @student select '二班' ,'B001','B4',86
insert @student select '二班' ,'B001','B5',88
insert @student select '二班' ,'B001','B6',99
insert @student select '二班' ,'B001','B7',100
insert @student select '二班' ,'B001','B8',90
insert @student select '二班' ,'B001','B9',97 

insert @tj select '一班',3
insert @tj select '二班',5

参考SQL:

-- 2005.T-SQL 
select t.bj,s.xh,s.name,s.cj
from @tj t
cross apply (
    SELECT TOP(t.num) 
            with ties       
-- 加 with ties,一班将选出4个人(2个人并列第三名) 
        xh,name,cj 
    from @student
    where t.bj=bj            
-- 加where 功能类似于 inner join ;不加类似于 cross join
    order by cj desc
)s
order by case when t.bj='一班' then 1 else 2 end asc,s.cj desc,s.xh asc    
---排序

本文转自:http://blog.jobbole.com/94919/

你真的会玩SQL吗?Top和Apply

本章预先想写一些Top和Apply基本的用法,但好像没什么意义,所以删掉了一些无用的东西,只留下几个示例,以保证系列的完整性。 Top和Apply解决的常见问题,如返回每个雇员的3个最新订单,订单的...
  • wxw_317
  • wxw_317
  • 2015年11月12日 16:20
  • 358

你真的会玩SQL吗?内连接、外连接

大多数人一般写多表查询会这样写select * from tbA ,tbB  没有用到JOIN关键字,太Low了,官网标准建议是用JOIN明确表间的关系,下面具体来讲。 连接类型: ...

你真的会玩SQL吗?实用函数方法汇总

实用函数方法 由于有些知识很少被用到,但真需要用时却忘记了又焦头烂额的到处找。 现在将这些‘冷门“却有效的小知识贡献出来,以备不时之需。 存储过程中的 '''' 相当于数...

SQL中ROW_NUMBER和APPLY在处理TOP N等类似问题的一点比较

[转自]http://www.cnblogs.com/changbluesky/archive/2010/07/19/1780593.html [由于本人是sql2000,不支持ROW_NUMBER...

【城会玩系列】hdu 5365 Run【计算几何相关】

Run Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total Subm...

听说你不会玩直播技术?

你不会,我免费教你

CVE-2016-5696漏洞分析:TCP侧信道安全 Leon不会玩

* 本文原创作者:Leon不会玩QEMU,本文属FreeBuf原创奖励计划,未经许可禁止转载 在这篇文章中,我们要讨论一个最新的TCP侧信道的漏洞(CVE-2016-5696)。这个标准是在...

真会玩!刘强东搞了个”猪脸识别“比赛,冠军奖单人能拿30万!

11月6日,随着首届“JDD-2017京东金融全球数据探索者大会”大幕拉开,备受关注的“JDD-2017京东金融全球数据探索者大赛”也终于揭晓,据小编了解,从即日起至11月20日,想参加JDD大赛的所...

Deeplink做不出效果,那是你不会玩!

根据我们的客户案例,再结合我们的使用经验和相关评估,下面我们提取出了一些Deeplink的进阶玩法。...

【城会玩】hdu 4920 Matrix multiplication【矩阵相乘优化】

Matrix multiplication Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:你真的会玩 SQL 吗?冷落的 Top 和 Apply
举报原因:
原因补充:

(最多只允许输入30个字)