2013-10-18 50 views
0

对于我的下一个把戏,我想只选择每个客户端的最近的事件。我想要一个,而不是四个000017的事件。TSQL OVER(PARTITION BY ...)

OK c_id e_date e_ser e_att e_recip Age c_cm e_staff rn 
--> 000017 2013-04-02 00:00:00.000 122 1 1 36 90510 90510 15 
--> 000017 2013-02-26 00:00:00.000 122 1 1 36 90510 90510 20 
--> 000017 2013-02-12 00:00:00.000 122 1 1 36 90510 90510 24 
--> 000017 2013-01-29 00:00:00.000 122 1 1 36 90510 90510 27 
--> 000188 2012-11-02 00:00:00.000 160 1 1 31 1289 1289 44 
--> 001713 2013-10-01 00:00:00.000 142 1 1 26 2539 2539 1 
--> 002531 2013-07-12 00:00:00.000 190 1 1 61 1689 1689 21 
--> 002531 2013-06-14 00:00:00.000 190 1 1 61 1689 1689 30 
--> 002531 2013-06-07 00:00:00.000 190 1 1 61 1689 1689 31 
--> 002531 2013-05-28 00:00:00.000 122 1 1 61 1689 1689 33 

这里是一个让我到这个阶段查询(也许你有一些建议,以改善这个问题,以及,多余的嵌套查询创建T2表可能是过度的。)谢谢大家!

SELECT TOP(10)* 
FROM (

    SELECT * 
    FROM (

    SELECT (SELECT CASE WHEN 
    (e_att IN (1,2) 
    AND e_date > DATEADD(month, -12, getdate()) 
    AND e_ser NOT IN (100,115) 
    AND e_recip NOT IN ('2','7') 
    AND (((e_recip = '3') AND (DATEDIFF(Year, c_bd, GetDate())>10)) OR (e_recip <> '3')) 
    AND c_cm = e_staff) 
    THEN '-->' 
    WHEN 1=1 THEN '' 
    END 
    ) AS 'OK' 
    ,c_id, e_date, e_ser, e_att, e_recip, DATEDIFF(Year, c_bd, GetDate()) AS 'Age', c_cm, e_staff 
    ,row_number() OVER (PARTITION BY c_id ORDER BY e_date DESC) rn    
    FROM events INNER JOIN client ON e_case_no = c_id 
    LEFT OUTER JOIN doc ON doc.doc_dbid = client.c_id 
    WHERE client.c_id IN (/* confidential query */) 
    AND e_date > DATEADD(month, -12, getdate()) 
    AND e_ser BETWEEN 11 AND 1000 
    GROUP BY  c_id, e_date, e_ser, e_att, e_recip, c_bd, c_cm, e_staff 
    ) t1 
) t2 
WHERE   OK = '-->' 
ORDER BY  c_id, e_date DESC 
+0

问题是什么? – McGarnagle

+0

嗨McGarnagle,如何为每个客户选择最近的事件?谢谢。 – PowderSnorkel

+0

我刚刚尝试添加另一个PARTITION BY到最外层选择,但我得到一个错误“无效的列名'rn2'。” – PowderSnorkel

回答

2

它看起来像下面产生的行号,按日期排序,每个客户端:

) t1 
    WHERE rn = 1 
) t2 
+0

甜!你达人! (或女性)嗯,这种评论可能会让一个人陷入麻烦。不管怎样,谢谢。 – PowderSnorkel

+1

@PowderSnorkel“男人”在这个网站上是一个相当安全的赌注,我想。 – McGarnagle

+0

:)周末愉快! – PowderSnorkel

0

,row_number() OVER (PARTITION BY c_id ORDER BY e_date DESC) rn    

因此增加where rn=1应该每个客户端产生最近的事件这里是你的原始查询了一些改进:

SELECT TOP(10) * 
FROM (

    SELECT '-->' AS 'OK' -- always this see where. 
    ,c_id, e_date, e_ser, e_att, e_recip, DATEDIFF(Year, c_bd, GetDate()) AS 'Age', c_cm, e_staff 
    ,row_number() OVER (PARTITION BY c_id ORDER BY e_date DESC) rn    
    FROM events INNER JOIN client ON e_case_no = c_id 
    LEFT OUTER JOIN doc ON doc.doc_dbid = client.c_id 
    WHERE client.c_id IN (/* confidential query */) 
      -- this part was in case and then filtered for later, if we put it in where now more efficient 
      (e_att IN (1,2) AND e_date > DATEADD(month, -12, getdate()) 
      AND e_ser NOT IN (100,115) 
      AND (((e_recip = '3') AND DATEDIFF(Year, c_bd, GetDate()>10)) OR e_recip NOT IN ('2', '3', '7')) 
      AND c_cm = e_staff) 


    AND e_date > DATEADD(month, -12, getdate()) 
    AND e_ser BETWEEN 11 AND 1000 
    GROUP BY  c_id, e_date, e_ser, e_att, e_recip, c_bd, c_cm, e_staff 
) t2 
ORDER BY  c_id, e_date DESC 

而且除去一些非必要的括号,如果从CASE语句移动的东西的,你不需要它过滤外部查询,这使得它更简单。

添加在ROW_NUMBER声明从McGarnagle的答案,你应该得到你想要的结果。

相关问题