這裏是一個很好的解決方案我剛剛發現。
選擇TOP N行的每個組 阿尼·羅蘭,2008年3月13日
有每個類別多行,並有一個願望, 選擇每個類別的只有前兩(2)行按價格。 例如,從以下數據:
RowID Category ID Description Price
1 Pot A1 Small Saucepan 21.50
2 Pot A2 1 Qt Saucepan 29.95
3 Pot A3 1.5 Qt Saucepan 33.95
4 Pot A4 Double Boiler 39.50
5 Pot A5 Stewpot 49.50
6 Pot A6 Pressure Cooker 79.95
7 Pan B1 8" Pie 6.95
8 Pan B2 8" Sq Cake 7.50
9 Pan B3 Bundt Cake 12.50
10 Pan B4 9x12 Brownie 7.95
11 Bowl C1 Lg Mixing 27.50
12 Bowl C2 Sm Mixing 17.50
13 Tools T1 14" Spatula 9.95
所需的輸出是:
RowID Category ID Description Price
11 Bowl C1 Lg Mixing 27.50
12 Bowl C2 Sm Mixing 17.50
9 Pan B3 Bundt Cake 12.50
10 Pan B4 9x12 Brownie 7.95
6 Pot A6 Pressure Cooker 79.95
5 Pot A5 Stewpot 49.50
13 Tools T1 14" Spatula 9.95
有幾種方法來完成所期望的輸出。 此演示提供的SQL Server 2005/SQL Server 2008中, ,然後SQL Server中的解決方案2000
兩種解決方案
創建示例數據
-- Suppress data loading messages
SET NOCOUNT ON
-- Create Sample Data using a Table Variable
DECLARE @MyTable table
( RowID int IDENTITY,
Category varchar(5),
[ID] varchar(5),
[Description] varchar(25),
Price decimal(10,2)
)
-- Load Sample Data
INSERT INTO @MyTable VALUES ('Pot', 'A1', 'Small Saucepan', 21.50)
INSERT INTO @MyTable VALUES ('Pot', 'A2', '1 Qt Saucepan', 29.95)
INSERT INTO @MyTable VALUES ('Pot', 'A3', '1.5 Qt Saucepan', 33.95)
INSERT INTO @MyTable VALUES ('Pot', 'A4', 'Double Boiler', 39.50)
INSERT INTO @MyTable VALUES ('Pot', 'A5', 'Stewpot', 49.50)
INSERT INTO @MyTable VALUES ('Pot', 'A6', 'Pressure Cooker', 79.95)
INSERT INTO @MyTable VALUES ('Pan', 'B1', '8"" Pie', 6.95)
INSERT INTO @MyTable VALUES ('Pan', 'B2', '8"" Sq Cake', 7.50)
INSERT INTO @MyTable VALUES ('Pan', 'B3', 'Bundt Cake', 12.50)
INSERT INTO @MyTable VALUES ('Pan', 'B4', '9x12 Brownie', 7.95)
INSERT INTO @MyTable VALUES ('Bowl', 'C1', 'Lg Mixing', 27.50)
INSERT INTO @MyTable VALUES ('Bowl', 'C2', 'Sm Mixing', 17.50)
INSERT INTO @MyTable VALUES ('Tools', 'T1', '14"" Spatula', 9.95)
Return to Top
的SQL Server 2005/SQL Server 2008中溶液的溶液
--Query to Retrieve Desired Data
SELECT
RowID,
Category,
[ID],
[Description],
Price
FROM (SELECT
ROW_NUMBER() OVER (PARTITION BY Category ORDER BY Price DESC) AS 'RowNumber',
RowID,
Category,
[ID],
[Description],
Price
FROM @MyTable
) dt
WHERE RowNumber <= 2
-- Results
RowID Category ID Description Price
11 Bowl C1 Lg Mixing 27.50
12 Bowl C2 Sm Mixing 17.50
9 Pan B3 Bundt Cake 12.50
10 Pan B4 9x12 Brownie 7.95
6 Pot A6 Pressure Cooker 79.95
5 Pot A5 Stewpot 49.50
13 Tools T1 14" Spatula 9.95
Return to Top
的SQL Server 2005/SQL服務器使用CTE 2008解決方案(添加人:雅各塞巴斯蒂安)
-- Define a CTE with the name "dt"
;WITH dt AS (
SELECT
ROW_NUMBER() OVER (PARTITION BY Category ORDER BY Price DESC) AS 'RowNumber',
RowID,
Category,
[ID],
[Description],
Price
FROM @MyTable
)
-- and select the data from the CTE
SELECT
RowID,
Category,
[ID],
[Description],
Price
FROM dt
WHERE RowNumber <= 2
-- Results
RowID Category ID Description Price
11 Bowl C1 Lg Mixing 27.50
12 Bowl C2 Sm Mixing 17.50
9 Pan B3 Bundt Cake 12.50
10 Pan B4 9x12 Brownie 7.95
6 Pot A6 Pressure Cooker 79.95
5 Pot A5 Stewpot 49.50
13 Tools T1 14" Spatula 9.95
Return to Top
SQL 2000解決方案
--Query to Retrieve Desired Data
SELECT DISTINCT
RowID,
Category,
[ID],
[Description],
Price
FROM @MyTable t1
WHERE RowID IN (SELECT TOP 2
RowID
FROM @MyTable t2
WHERE t2.Category = t1.Category
ORDER BY Price DESC
)
ORDER BY
Category,
Price DESC
-- Results
RowID Category ID Description Price
11 Bowl C1 Lg Mixing 27.50
12 Bowl C2 Sm Mixing 17.50
9 Pan B3 Bundt Cake 12.50
10 Pan B4 9x12 Brownie 7.95
6 Pot A6 Pressure Cooker 79.95
5 Pot A5 Stewpot 49.50
13 Tools T1 14" Spatula 9.95
來源:Select the TOP n Rows For Each Group
什麼SQL引擎?不幸的是,SQL標準,PostgreSQL,MS SQL Server,Oracle,IBM DB2等等都有一種非常好的方式來完成你想要的任務 - 但是如果你堅持使用MySQL,那麼這個完美的解決方案每一個好的關係數據庫並且根據標準本身是不可用的,所以它是非常快速的(與MySQL有關的課程 - 嘆息)。那麼它將會是什麼樣子呢 - 這個星球上的每一個體面的SQL實現,或者另一方面的MySQL,或者其他......? – 2009-08-30 04:49:22
@Alex:在叢林中跳動 - 讓我們知道您對MySQL的感受!不要讓它瓶裝。 ;-) – 2009-08-30 04:51:36
mySQL> MS Access ...勉強 – 2009-08-30 05:05:01