У меня около 100 таблиц, где все они имеют дублирующие ограничения внешнего ключа на них.удаление дубликатов внешних ключей
Есть ли способ, чтобы я мог избавиться от него? Есть ли запрос, который может дать мне все дублирующие ключи?
У меня около 100 таблиц, где все они имеют дублирующие ограничения внешнего ключа на них.удаление дубликатов внешних ключей
Есть ли способ, чтобы я мог избавиться от него? Есть ли запрос, который может дать мне все дублирующие ключи?
Я использую этот сценарий T-SQL здесь, чтобы обнаружить потенциально дублирующие ограничения FK - и он также создает необходимые операторы ALTER TABLE...DROP CONSTRAINT
в последнем из столбцов вывода.
Вы не можете надежно автоматически обнаруживать и выбирать, какие из множества ограничений FK нужно отбрасывать, - поэтому вы, в основном, можете их обнаружить, а затем вручную выбрать, какие из них следует отбрасывать (используя этот оператор сброса, созданный моим запросом).
;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
В течение 100 таблиц это не вариант, но если у вас есть только несколько таблиц, создать диаграмму в SQL Server Management Studio, добавлять таблицы там и удалить дубликаты визуально.
Это удаляет последних созданных дубликаты
;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: ** SURE! ** обновил мой ответ, включив родительскую схему (вы также можете получить ссылочную схему, если это необходимо, но вам это не нужно для оператора DROP) –