วิธีรับแถว MAX


20

ใน SQL Server ฉันพบว่ามันเป็นความเจ็บปวดที่จะได้รับแถวสูงสุดสำหรับชุดข้อมูลฉันกำลังมองหารายการของวิธีการดึงแถวสูงสุดด้วยคำแนะนำเกี่ยวกับประสิทธิภาพและการบำรุงรักษา

ตารางตัวอย่าง:

DECLARE @Test TABLE (ID INT IDENTITY(1,1), name VARCHAR(50), 
                     dateOfBirth DATETIME, TaxNumber varchar(10))

INSERT INTO @Test (name, dateOfBirth, TaxNumber)
SELECT 'Fred', convert(datetime, '25/01/1976', 103), '123' UNION ALL
SELECT 'Bob', convert(datetime, '03/03/1976', 103), '234'  UNION ALL
SELECT 'Jane', convert(datetime, '13/06/1996', 103), '345' UNION ALL
SELECT 'Fred', convert(datetime, '14/02/1982', 103), '456' UNION ALL
SELECT 'Bob', convert(datetime, '25/10/1983', 103), '567' UNION ALL
SELECT 'Jane', convert(datetime, '12/04/1995', 103), '678' UNION ALL
SELECT 'Fred', convert(datetime, '03/03/1976', 103), '789'

select * from @Test

ให้:

ID          name      dateOfBirth             TaxNumber
----------- --------- ----------------------- ----------
1           Fred      1976-01-25 00:00:00.000 123
2           Bob       1976-03-03 00:00:00.000 234
3           Jane      1996-06-13 00:00:00.000 345
4           Fred      1982-02-14 00:00:00.000 456
5           Bob       1983-10-25 00:00:00.000 567
6           Jane      1995-04-12 00:00:00.000 678
7           Fred      1976-03-03 00:00:00.000 789

หากฉันต้องการดึงรายละเอียดที่เก่าที่สุด (จัดกลุ่มตามชื่อ) รายละเอียดฉันจะใช้วิธีการใดได้บ้าง?

ผลลัพธ์ที่ต้องการ:

ID          name      dateOfBirth             TaxNumber
----------- --------- ----------------------- ----------
1           Fred      1976-01-25 00:00:00.000 123
2           Bob       1976-03-03 00:00:00.000 234
6           Jane      1995-04-12 00:00:00.000 678

คำตอบ:


20

สองวิธีปกติ: ฟังก์ชันการรวมและการจัดอันดับ

ผลรวมจะทำงานกับ SQL Server 2000 ทั้งสองวิธีสามารถใช้ CTE หรือตารางที่ได้รับ

เพื่อประสิทธิภาพฉันพบว่าผลรวมนั้นทำงานได้ดีขึ้น อย่างไรก็ตามดูเหมือนว่าฟังก์ชั่นการจัดอันดับของ SQL Server 2008 นั้นทำงานได้ดีกว่าใน SQL Server 2005 ฉันยังไม่ได้ใช้ SQL Server 2008 ทุกวัน

มี 2 ​​คำถาม SO ที่เกี่ยวข้อง แต่ฉันไม่สามารถค้นหาได้ในขณะนี้ หนึ่งคือคำถามเกี่ยวกับ IO ตรรกะสูงที่มีฟังก์ชั่นการจัดอันดับอีกประการหนึ่งคือการทดสอบการจัดอันดับในความคิดเห็นมากกว่า SQL 2k5 เทียบกับ 2k8 ขอโทษ

--aggregate + CTE
;WITH cOldest AS
(
    SELECT name, MIN(dateOfBirth) AS MinDOB FROM @Test GROUP BY name
)
SELECT
    T.*
FROM
    @Test T
    JOIN
    cOldest C ON T.name = C.name AND T.dateOfBirth = C.MinDOB
ORDER BY
    T.ID

--aggregate + derived table
SELECT
    T.*
FROM
    @Test T
    JOIN
    (
    SELECT name, MIN(dateOfBirth) AS MinDOB FROM @Test GROUP BY name
    ) C ON T.name = C.name AND T.dateOfBirth = C.MinDOB
ORDER BY
    T.ID

--ranking + CTE
;WITH cOldest AS
(
    SELECT *, ROW_NUMBER() OVER (PARTITION BY name ORDER BY dateOfBirth) AS rnDOB FROM @Test
)
SELECT
    C.*
FROM
    cOldest C
WHERE
    C.rnDOB = 1
ORDER BY
    C.ID

--ranking + derived table
SELECT
    C.*
FROM
    (SELECT *, ROW_NUMBER() OVER (PARTITION BY name ORDER BY dateOfBirth) AS rnDOB FROM @Test) C
WHERE
    C.rnDOB = 1
ORDER BY
    C.ID
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.