SQL显示其他类别中不存在的项目

kev*_*vro 0 sql t-sql sql-server sql-server-2012

我有一个Tier,Category和Items的表.对于Tier = 2,每个类别应具有相同的项目数.我想显示每个类别中缺少的项目.

这是一个示例表:

IF OBJECT_ID('tempdb..#Items') IS NOT NULL DROP TABLE #Items
CREATE TABLE #Items (
    Tier        int,
    Category    nvarchar(25),
    Item        nvarchar(25)
)

INSERT INTO #Items
SELECT '2', 'CategoryA',    'Item1'
UNION ALL
SELECT '2', 'CategoryA',    'Item2'
UNION ALL
SELECT '2', 'CategoryA',    'Item3'
UNION ALL
SELECT '2', 'CategoryB',    'Item1'
UNION ALL
SELECT '2', 'CategoryB',    'Item3'
UNION ALL
SELECT '2', 'CategoryC',    'Item1'
UNION ALL
SELECT '2', 'CategoryC',    'Item2'
UNION ALL
SELECT '2', 'CategoryC',    'Item3'
UNION ALL
SELECT '2', 'CategoryC',    'Item4'

SELECT
    Tier,
    Category,
    Item
FROM #Items
ORDER BY Tier, Category, Item

Tier    Category    Item
2   CategoryA   Item1
2   CategoryA   Item2
2   CategoryA   Item3
2   CategoryB   Item1
2   CategoryB   Item3
2   CategoryC   Item1
2   CategoryC   Item2
2   CategoryC   Item3
2   CategoryC   Item4
Run Code Online (Sandbox Code Playgroud)

如果我显示每个类别中的项目数:

SELECT
    Tier,
    Category,
    COUNT(Item) as ItemCount
FROM #Items
GROUP BY Tier, Category
Run Code Online (Sandbox Code Playgroud)

他们应该都有相同的4项:

Tier    Category    ItemCount
2   CategoryA   3
2   CategoryB   2
2   CategoryC   4
Run Code Online (Sandbox Code Playgroud)

我想显示每个类别中缺少的项目列表.这可能有效,但显然是笨拙的,只是为了呈现数据我想看到它.

SELECT DISTINCT
    Tier,
    'CategoryA' as [Category],
    Item
FROM #Items i
WHERE Item NOT IN (SELECT Item from #Items WHERE Category = 'CategoryA')
UNION ALL
SELECT DISTINCT
    Tier,
    'CategoryB' as [Category],
    Item
FROM #Items i
WHERE Item NOT IN (SELECT Item from #Items WHERE Category = 'CategoryB')
UNION ALL
SELECT DISTINCT
    Tier,
    'CategoryC' as [Category],
    Item
FROM #Items i
WHERE Item NOT IN (SELECT Item from #Items WHERE Category = 'CategoryC')
Run Code Online (Sandbox Code Playgroud)

并显示结果:

Tier    Category    Item
2   CategoryA   Item4
2   CategoryB   Item2
2   CategoryB   Item4
Run Code Online (Sandbox Code Playgroud)

D S*_*ley 5

您可以使用项目的笛卡尔积来完成,然后拉出源表中不存在的项:

SELECT DISTINCT
    i1.Tier,
    i1.[Category],
    i2.Item
FROM Items i1, Items i2
WHERE i2.Item NOT IN (SELECT Item from Items WHERE Category = i1.Category)
Run Code Online (Sandbox Code Playgroud)

结果:

TIER    CATEGORY    ITEM
----    --------    -----
2       CategoryA   Item4
2       CategoryB   Item2
2       CategoryB   Item4
Run Code Online (Sandbox Code Playgroud)

您可以在此处测试查询.