MW211 EXIT

devlog
MSSQL/対等外部結合
2015年04月22日
外部結合をつかって、二つの表を対等外部結合してみた。
┌──────────────────────────────────────┐
│SELECT ISNULL([表A].[キー],[表B].[キー]) AS [キー],                       │
│       [表A].[項目A],                                                     │
│       [表B].[項目B]                                                      │
│    FROM [表A]                                                             │
│        FULL JOIN [表B]                                                    │
│          ON [表B].[キー] = [表A].[キー]                                  │
└──────────────────────────────────────┘
結果は上々で、以下三パターンのレコードとなった(もちろんキーの重複はなし)。
┌──┬───┬───┐
│キー│項目A│項目B│
├──┼───┼───┤
│キー│項目A│  -  │
├──┼───┼───┤
│キー│  -  │項目B│
└──┴───┴───┘

これに気を良くして、三つの表でこれを試みた。
┌──────────────────────────────────────┐
│SELECT ISNULL([表A].[キー],ISNULL([表B].[キー],[表C].[キー])) AS [キー], │
│       [表A].[項目A],                                                     │
│       [表B].[項目B],                                                     │
│       [表C].[項目C]                                                      │
│    FROM [表A]                                                             │
│        FULL JOIN [表B]                                                    │
│          ON [表B].[キー] = [表A].[キー]                                  │
│        FULL JOIN [表C]                                                    │
│          ON [表C].[キー] = [表A].[キー]                                  │
└──────────────────────────────────────┘
すると基幹となる表がない場合、二つに分裂してしまった。
┌──┬───┬───┬───┐
│キー│項目A│項目B│項目C│
├──┼───┼───┼───┤
│キー│項目A│項目B│  -  │
├──┼───┼───┼───┤
│キー│項目A│  -  │項目C│
├──┼───┼───┼───┤
│キー│項目A│  -  │  -  │
├──┼───┼───┼───┤
│キー│  -  │項目B│項目C│
├──┼───┼───┼───┤
│キー│  -  │項目B│  -  │←二件レコードができてしまう
│キー│  -  │  -  │項目C│
└──┴───┴───┴───┘

なので、トーナメントの二段階選抜的な形にしなければならない。
┌──────────────────────────────────────┐
│SELECT ISNULL([連合D].[キー],[表C].[キー]) AS [キー],                     │
│       [連合D].[項目A],                                                   │
│       [連合D].[項目B],                                                   │
│       [表C].[項目C]                                                      │
│    FROM (                                                                  │
│        SELECT ISNULL([表A].[キー],[表B].[キー]) AS [キー],               │
│               [表A].[項目A],                                             │
│               [表B].[項目B]                                              │
│            FROM [表A]                                                     │
│                FULL JOIN [表B]                                            │
│                  ON [表B].[キー] = [表A].[キー]                          │
│    )AS [連合D]                                                            │
│        FULL JOIN [表C]                                                    │
│          ON [表C].[キー] = [連合D].[キー]                                │
└──────────────────────────────────────┘

でも、これだと四つ以上になった時に複雑度が増していきそうなので
(でもそれこそトーナメント式(B木)で書けたりして(でもいずれにせよ煩雑になりそう))
以下の様に、基準データをあらかじめ決めてから、
そこに左外部結合していくのがいいかも。
┌──────────────────────────────────────┐
│SELECT [基準D].[キー],                                                     │
│       [表A].[項目A],                                                     │
│       [表B].[項目B],                                                     │
│       [表C].[項目C]                                                      │
│    FROM (                                                                  │
│        SELECT [キー] FROM [表A]                                           │
│        UNION                                                               │
│        SELECT [キー] FROM [表B]                                           │
│        UNION                                                               │
│        SELECT [キー] FROM [表C]                                           │
│    ) AS [基準D]                                                           │
│        LEFT JOIN [表A]                                                    │
│          ON [表A].[キー] = [基準D].[キー]                                │
│        LEFT JOIN [表B]                                                    │
│          ON [表B].[キー] = [基準D].[キー]                                │
│        LEFT JOIN [表C]                                                    │
│          ON [表C].[キー] = [基準D].[キー]                                │
└──────────────────────────────────────┘
分類:MSSQL