私はそれらのすべてに重複する外部キー制約がある約100のテーブルを持っています。重複した外部キーを削除する
私はそれを取り除く方法がありますか?私にすべての重複キーを与えることができるクエリはありますか?
私はそれらのすべてに重複する外部キー制約がある約100のテーブルを持っています。重複した外部キーを削除する
私はそれを取り除く方法がありますか?私にすべての重複キーを与えることができるクエリはありますか?
ここでは、このT-SQLスクリプトを使用して、潜在的に重複するFK制約を検出します。また、最後の出力列に必要なALTER TABLE...DROP CONSTRAINT
文を生成します。
ドロップする複数のFK制約のうち、どれを確実に検出して選択することはできません - 基本的にはそれらを検出して、ドロップするものを手動で選択します(クエリによって生成されたdropステートメントを使用します)。 100テーブルの場合
;WITH FKData AS
(
SELECT
fk.parent_object_id,
fkc.parent_column_id,
fk.referenced_object_id,
fkc.referenced_column_id,
FKCount = COUNT(*)
FROM
sys.foreign_keys fk
INNER JOIN
sys.foreign_key_columns fkc ON fkc.constraint_object_id = fk.object_id
GROUP BY
fk.parent_object_id, fkc.parent_column_id, fk.referenced_object_id, fkc.referenced_column_id
HAVING
COUNT(*) > 1
),
DuplicateFK AS
(
SELECT
FKName = fk.Name,
ParentSchema = s1.Name,
ParentTable = t1.Name,
ParentColumn = c1.Name,
ReferencedTable = t2.Name,
ReferencedColumn = c2.Name
FROM
sys.foreign_keys fk
INNER JOIN
sys.foreign_key_columns fkc ON fkc.constraint_object_id = fk.object_id
INNER JOIN
FKData f ON fk.parent_object_id = f.parent_object_id
AND fk.referenced_object_id = f.referenced_object_id
AND fkc.parent_column_id = f.parent_column_id
AND fkc.referenced_column_id = f.referenced_column_id
INNER JOIN
sys.tables t1 ON f.parent_object_id = t1.object_id
INNER JOIN
sys.columns c1 ON f.parent_object_id = c1.object_id AND f.parent_column_id = c1.column_id
INNER JOIN
sys.schemas s1 ON t1.schema_id = s1.schema_id
INNER JOIN
sys.tables t2 ON f.referenced_object_id = t2.object_id
INNER JOIN
sys.columns c2 ON f.referenced_object_id = c2.object_id AND f.referenced_column_id = c2.column_id
)
SELECT
FKName,
ParentSchema, ParentTable, ParentColumn,
ReferencedTable, ReferencedColumn,
DropStmt = 'ALTER TABLE ' + ParentSchema + '.' + ParentTable +
' DROP CONSTRAINT ' + FKName
FROM
DuplicateFK
は、それはオプションではありませんが、あなただけのいくつかのテーブルを持っている場合は、SQL Server Management Studioでダイアグラムを作成するが、あなたのテーブルを追加し、視覚的にdupesを削除します。
これは素晴らしい最近作成した重複
;WITH fkeys AS (
SELECT f.object_id ,
f.name ,
f.parent_object_id,
ROW_NUMBER() OVER(PARTITION BY t.column_names ORDER BY f.create_date,f.[object_id]) AS RowNum
FROM sys.foreign_keys f
CROSS APPLY (SELECT fc.parent_object_id,parent_column_id,fc.referenced_object_id ,fc.referenced_column_id
FROM sys.foreign_key_columns fc
WHERE fc.constraint_object_id = f.object_id
ORDER BY constraint_column_id
FOR XML PATH('')
) t (column_names)
)
SELECT 'ALTER TABLE '+QUOTENAME(OBJECT_SCHEMA_NAME(f.parent_object_id)) + '.'+QUOTENAME(OBJECT_NAME(f.parent_object_id)) +' DROP CONSTRAINT '+QUOTENAME(f.name)+';' AS DropStatement
FROM fkeys f
WHERE f.RowNum >= 2
サンキュー削除!! – peter
は、スキーマ名を取得する方法もありますか? dboスキーマに属していないものはほとんどないからです。 – peter
@Peter:**確かに!**親スキーマを含むように私の応答を更新しました(必要ならば参照スキーマも取得できますが、DROPステートメントには必要ありません) –