LoginSignup
0
0

More than 5 years have passed since last update.

SQLServer: 一列目のカラムが重複しているインデックスを探すSQL

Posted at

ディスク容量を減らしたいときや、更新性能UPのために不要なインデックスの候補を見つける方法です。

未使用インデックスの調査とは別に、既存のインデックスでひとまとめにできたり削除できるインデックスがないかを調べたいと思い、以下のクエリを作成しました。

一列目のカラムが重複しているインデックスは、場合によってはまとめられたり片方削除しても大丈夫だよね、という発想です。
最終的には開発者の判断が必要ですが、実際にこのクエリをベースにして多少のディスク容量削減を実現できました。

SELECT
   mainO.name as tableName --テーブル名
  ,mainO.object_id as tableID
  ,mainI.name as indexName --インデックス名
  ,mainI.index_id --0:heap 1:clusterd >1:nonclustered
  ,mainI.type --0:heap 1:clustered 2:nonclustered >3:詳細は別途referenceを参照
  ,mainI.is_unique
  ,mainI.is_primary_key
  ,mainI.is_unique_constraint
  ,mainIC.index_column_id --インデックス内での固有のID
  ,mainIC.column_id --カラムと一対一に対応したID
  ,COL_NAME(mainIC.object_id, mainIC.column_id) as dup_column_name --重複カラム
  ,mainIC.key_ordinal --インデックス内でのキーの並び順
  ,mainIC.is_included_column --1:付加列 0:付加列じゃない
  ,mainIC.is_descending_key --1:降順 0:昇順
  ,mainPS.reserved_page_count * 8.0 / 1024 as 'Size(MB)'
FROM
  sys.objects AS mainO WITH (NOLOCK)
JOIN sys.indexes AS mainI WITH (NOLOCK) ON mainO.object_id = mainI.object_id
JOIN sys.index_columns AS mainIC ON mainI.object_id = mainIC.object_id AND mainI.index_id = mainIC.index_id
LEFT JOIN sys.dm_db_partition_stats AS mainPS ON mainPS.object_id = mainI.object_id and mainPS.index_id = mainI.index_id
WHERE
EXISTS (
  SELECT
    *
  FROM
  (
    SELECT
      o.name as tableName --テーブル名
      ,o.object_id as tableID
      ,i.name as indexName --インデックス名
      ,i.index_id --0:heap 1:clusterd >1:nonclustered
      ,i.type --0:heap 1:clustered 2:nonclustered >3:詳細は別途referenceを参照
      ,i.is_unique
      ,i.is_primary_key
      ,i.is_unique_constraint
      ,ic.index_column_id --インデックス内での固有のID
      ,ic.column_id --カラムと一対一に対応したID
      ,COL_NAME(ic.object_id, ic.column_id) as column_name
      ,ic.key_ordinal --インデックス内でのキーの並び順
      ,ic.is_included_column --1:付加列 0:付加列じゃない
      ,ic.is_descending_key --1:降順 0:昇順
    FROM
      sys.objects AS o WITH (NOLOCK)
    JOIN sys.indexes AS i WITH (NOLOCK) ON o.object_id = i.object_id
    JOIN sys.index_columns AS ic ON i.object_id = ic.object_id AND i.index_id = ic.index_id
    WHERE
      o.type = 'U'
    AND i.name NOT LIKE '_WA_Sys_%'
    --AND i.object_id = OBJECT_ID('テーブル名'); 
  ) AS A
  WHERE
    key_ordinal = 1 --第一列に限定
  GROUP BY
    tableID
    ,tableName
    ,column_id
  HAVING
    COUNT(*) > 1 --同一テーブル内で、第一列に同じカラムを指定したインデックスが複数あるものに限定
  AND tableID = mainO.object_id
  AND column_id = mainIC.column_id
)
AND key_ordinal = 1
AND (mainPS.reserved_page_count * 8.0 / 1024) > 500 --500MB以上のものに限定
ORDER BY
  mainO.name
  ,mainI.index_id
  ,mainIC.column_id

単純にテーブル、インデックス、インデックス内のカラム一覧をリストアップするクエリも需要あるかなと思いまして、のせておきます。

-- テーブル、インデックス、インデックス内のカラム一覧
SELECT
  o.name --テーブル名
  ,o.object_id
  ,i.name --インデックス名
  ,i.index_id --0:heap 1:clusterd >1:nonclustered
  ,i.type --0:heap 1:clustered 2:nonclustered >3:詳細は別途referenceを参照
  ,i.is_unique
  ,i.is_primary_key
  ,i.is_unique_constraint
  ,ic.index_column_id --インデックス内での固有のID
  ,ic.column_id --カラムと一対一に対応したID
  ,COL_NAME(ic.object_id, ic.column_id) as column_name
  ,ic.key_ordinal --インデックス内でのキーの並び順
  ,ic.is_included_column --1:付加列 0:付加列じゃない
  ,ic.is_descending_key --1:降順 0:昇順
FROM
  sys.objects AS o WITH (NOLOCK)
JOIN sys.indexes AS i WITH (NOLOCK) ON o.object_id = i.object_id
JOIN sys.index_columns AS ic ON i.object_id = ic.object_id AND i.index_id = ic.index_id
WHERE
  o.type = 'U'
AND i.name NOT LIKE '_WA_Sys_%'
--AND i.object_id = OBJECT_ID('テーブル名'); 
ORDER BY
  o.name
  ,i.index_id
  ,ic.index_column_id
0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0