Sibainu Relax Room

柴犬と過ごす

SQL 連番を振る

神社の狛犬と競い合っている柴犬でした。

10歳になっても頑張るぞと言っているようです。

さて、今回は連番を振るSQLをACCESSで考えてみました。

テーブルとデータ

まず、テーブルの構造とデータは次のようなものを考えます。

次に、データです。

このデータを条件 code = ‘030’ で抽出して、名前の小さい順に並べ替え、1 から順に番号(順番)を振ります。

クエリーの作成・実行

次のようなクエリーをデザインビューではなくSQLビューで書きます。

というのは、デザインビューでは作成できません。

サブクエリーが幾重に入れ子になって、複雑な SQL 文になりました。

早速、実行してみました。

無事、想定の結果が得られました。

SQL

copy

SELECT
   Y.*
FROM
   (SELECT
      A.ID
      , A.code
      , A.名前
      , A.住所
      , (SELECT
            Count(*)
         FROM
            (SELECT
                C.*
             FROM
                sample AS C
             WHERE
                C.CODE='030') AS B
         WHERE
            A.名前 >= B.名前) AS 順番
   FROM
      (SELECT
          D.*
       FROM
          sample AS D
       WHERE
          D.CODE='030') AS A) AS Y
ORDER BY Y.順番;

名前がユニークな集合でない場合

上のデータは名前がユニークな集合ですが、次にユニークではない集合、たとえば次の例を考えます。

このデータに上と同じクエリーを実行してみますと、次のようになります。

やはり、「京都 三郎」は 1 としたいです。

クエリーの改良

それでは、「京都 三郎」は 1 、「京都 六郎」は 5 としたい場合、どうしたらよいのか考えてみます。

copy

SELECT Y.*
FROM (SELECT
      A.ID
      , A.code
      , A.名前
      , A.住所
      , (SELECT
            Count(*) + 1
         FROM
            (SELECT
                C.*
             FROM
                sample AS C
             WHERE
                C.CODE='030') AS B
         WHERE
            A.名前 > B.名前) AS 順番
   FROM
      (SELECT
          D.*
       FROM
          sample AS D
       WHERE
          D.CODE='030') AS A)  AS Y
ORDER BY Y.順番;

この SQL は、「 Count(*) 」を「 Count(*) + 1 」に、「 A.名前 >= B.名前 」から「 A.名前 > B.名前 」に変更しただけです。

変更した SQL の実行

早速、実行してみます。

想定した結果が得られました。

同一人の複数の表示が気になる

「京都 三郎」順位 1 が4レコード、「京都 六郎」順位 5 が2レコードの表示があるのを、同一人であるから1レコードにしたい。

名前の集合をユニークにすればよいので、名前をグループ化します。

copy

SELECT Y.*
FROM (SELECT
      A.ID
      , A.code
      , A.名前
      , A.住所
      , (SELECT
            Count(*) + 1
         FROM
            (SELECT
                MAX(C.ID) AS ID
                , MAX(C.code) AS code
                , C.名前
                , MAX(C.住所) AS 住所
             FROM
                sample AS C
             WHERE
                C.CODE='030'
             GROUP BY C.名前) AS B
         WHERE
            A.名前 > B.名前) AS 順番
   FROM
      (SELECT
          MAX(D.ID) AS ID
          , MAX(D.code) AS code
          , D.名前
          , MAX(D.住所) AS 住所
       FROM
          sample AS D
       WHERE
          D.CODE='030'
       GROUP BY D.名前) AS A)  AS Y
ORDER BY Y.順番;

実行

これを実行してみます。

同性同名に対応する

同性同名の別人があるデータを考えてみます。

次のデータでは、IDが 11 と 13 の「京都 三郎」の住所はそれぞれ「東京」「京都」で同性同名の別人です。

このような場合の SQL を考えてみます。

ユニークなデータの比較が出来ればよいはずです。

名前と住所を結合してユニークな集合を作り比較することにします。

copy

SELECT Y.*
FROM (SELECT
      A.ID
      , A.code
      , A.名前
      , A.住所
      , (SELECT
            Count(*) + 1
         FROM
            (SELECT
                MAX(C.ID) AS ID
                , MAX(C.CODE) AS code
                , MAX(C.名前) AS 名前
                , MAX(C.住所) AS 住所
                , C.名前住所
             FROM
                (SELECT
                    E.*
                    , E.名前 & E.住所 AS 名前住所
                 FROM
                    sample AS E
                 WHERE
                    E.CODE='030') AS C 
             GROUP BY C.名前住所) AS B
         WHERE
            A.名前住所 > B.名前住所) AS 順番
   FROM
      (SELECT
          MAX(D.ID) AS ID
          , MAX(D.CODE) AS code
          , MAX(D.名前) AS 名前
          , MAX(D.住所) AS 住所
          , D.名前住所
       FROM
          (SELECT
              F.*
              , F.名前 & F.住所 AS 名前住所
           FROM
              sample AS F
           WHERE
              F.CODE='030') AS D
       GROUP BY D.名前住所) AS A)  AS Y
ORDER BY Y.順番;

かなり長文の SQL になりましたが実行してみます。

ところで、MAX 関数を使ったのは、ORDER BY 句の記述を簡単にしたいからです。

これも想定とおりの結果が得られました。