繁体   English   中英

SQL 左加入最新记录

[英]SQL left join with latest record

我想离开加入一个只有最新记录的表。 我有Customer1表:

+--------+----------+
| CustID | CustName |
+--------+----------+
|      1 | ABC123   |
|      2 | 456XYZ   |
|      3 | 5PQR3    |
|      4 | 789XYZ   |
|      5 | 789A     |
+--------+----------+

SalesInvoice表:

+------------+--------+-----------+
|  InvDate   | CustID | InvNumber |
+------------+--------+-----------+
| 2020-03-01 |      1 | IV236     |
| 2020-04-07 |      1 | IV644     |
| 2020-06-13 |      2 | IV869     |
| 2020-03-29 |      3 | IV436     |
| 2020-02-06 |      3 | IV126     |
+------------+--------+-----------+

我想要这个必需的 output:

+--------+------------+-----------+
| CustID |  InvDate   | InvNumber |
+--------+------------+-----------+
|      1 | 2020-04-07 | IV644     |
|      2 | 2020-06-13 | IV869     |
|      3 | 2020-03-29 | IV436     |
|      4 |            |           |
|      5 |            |           |
+--------+------------+-----------+

为了方便快捷,下面是示例代码。

    drop table if exists #Customer1
    create table #Customer1(CustID int, CustName varchar (100))
    insert into #Customer1 values
    (1,'ABC123'),
    (2,'456XYZ'),
    (3,'5PQR3'),
    (4,'789XYZ'),
    (5,'789A')
    
    
    drop table if exists #SalesInvoice
    create table #SalesInvoice(InvDate DATE, CustID INT, InvNumber varchar (100))
    insert into #SalesInvoice values
    ('2020-03-01',1,'IV236'),
    ('2020-04-07',1,'IV644'),
    ('2020-06-13',2,'IV869'), 
    ('2020-03-29',3,'IV436'),
    ('2020-02-06',3,'IV126')

在这种情况下,我喜欢使用TOP 1 WITH TIES

SELECT TOP 1 WITH TIES c.CustID, i.InvDate, i.InvNumber
FROM #Customer1 c
LEFT JOIN #Invoices i ON c.CustID = i.CustID
ORDER BY ROW_NUMBER() OVER (PARTITION BY c.CustID ORDER BY i.InvDate DESC);

下面演示链接的屏幕截图

演示

这里的首要技巧是按行号排序,为每个客户分配一个序列,序列按发票日期递减。 然后,这种方法只保留每个客户的最新发票记录。

我推荐outer apply

select c.*, i.*
from #c c outer apply
     (select top (1) i.*
      from #invoices i
      where i.custId = c.custId
      order by i.invDate desc
     ) i;

outer apply实现了一种特殊类型的join ,称为“横向连接”。 这是一个非常强大的构造。 但是在了解它们时,您可以将横向连接视为可以返回多于一列和多于一行的相关子查询。

您可以尝试使用 ROW_NUMBER window function 而不是使用这个简单的自解释 T-SQL 进行横向连接

SELECT      c.CustID
            , d.InvDate
            , d.InvNumber
FROM        #C c
LEFT JOIN   (
            SELECT  *
                    , ROW_NUMBER() OVER (PARTITION BY CustID ORDER BY InvDate DESC) AS RowNo
            FROM    #D
            ) d
ON          c.CustID = d.CustID
            AND d.RowNo = 1

基本上,ROW_NUMBER 用于在一次表扫描中过滤“最后一张”发票,而不是在必须多次执行的相关查询中执行 SELECT TOP 1... ORDER BY - 与客户数量一样多。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM