How can foreign key constraints be temporarily disabled using T-SQL?

后端 未结 16 1940
Happy的楠姐
Happy的楠姐 2020-11-22 04:57

Are disabling and enabling foreign key constraints supported in SQL Server? Or is my only option to drop and then re-create

16条回答
  •  旧巷少年郎
    2020-11-22 05:45

    First post :)

    For the OP, kristof's solution will work, unless there are issues with massive data and transaction log balloon issues on big deletes. Also, even with tlog storage to spare, since deletes write to the tlog, the operation can take a VERY long time for tables with hundreds of millions of rows.

    I use a series of cursors to truncate and reload large copies of one of our huge production databases frequently. The solution engineered accounts for multiple schemas, multiple foreign key columns, and best of all can be sproc'd out for use in SSIS.

    It involves creation of three staging tables (real tables) to house the DROP, CREATE, and CHECK FK scripts, creation and insertion of those scripts into the tables, and then looping over the tables and executing them. The attached script is four parts: 1.) creation and storage of the scripts in the three staging (real) tables, 2.) execution of the drop FK scripts via a cursor one by one, 3.) Using sp_MSforeachtable to truncate all the tables in the database other than our three staging tables and 4.) execution of the create FK and check FK scripts at the end of your ETL SSIS package.

    Run the script creation portion in an Execute SQL task in SSIS. Run the "execute Drop FK Scripts" portion in a second Execute SQL task. Put the truncation script in a third Execute SQL task, then perform whatever other ETL processes you need to do prior to attaching the CREATE and CHECK scripts in a final Execute SQL task (or two if desired) at the end of your control flow.

    Storage of the scripts in real tables has proven invaluable when the re-application of the foreign keys fails as you can select * from sync_CreateFK, copy/paste into your query window, run them one at a time, and fix the data issues once you find ones that failed/are still failing to re-apply.

    Do not re-run the script again if it fails without making sure that you re-apply all of the foreign keys/checks prior to doing so, or you will most likely lose some creation and check fk scripting as our staging tables are dropped and recreated prior to the creation of the scripts to execute.

    ----------------------------------------------------------------------------
    1)
    /*
    Author:         Denmach
    DateCreated:    2014-04-23
    Purpose:        Generates SQL statements to DROP, ADD, and CHECK existing constraints for a 
                    database.  Stores scripts in tables on target database for execution.  Executes
                    those stored scripts via independent cursors. 
    DateModified:
    ModifiedBy
    Comments:       This will eliminate deletes and the T-log ballooning associated with it.
    */
    
    DECLARE @schema_name SYSNAME; 
    DECLARE @table_name SYSNAME; 
    DECLARE @constraint_name SYSNAME; 
    DECLARE @constraint_object_id INT; 
    DECLARE @referenced_object_name SYSNAME; 
    DECLARE @is_disabled BIT; 
    DECLARE @is_not_for_replication BIT; 
    DECLARE @is_not_trusted BIT; 
    DECLARE @delete_referential_action TINYINT; 
    DECLARE @update_referential_action TINYINT; 
    DECLARE @tsql NVARCHAR(4000); 
    DECLARE @tsql2 NVARCHAR(4000); 
    DECLARE @fkCol SYSNAME; 
    DECLARE @pkCol SYSNAME; 
    DECLARE @col1 BIT; 
    DECLARE @action CHAR(6);  
    DECLARE @referenced_schema_name SYSNAME;
    
    
    
    --------------------------------Generate scripts to drop all foreign keys in a database --------------------------------
    
    IF OBJECT_ID('dbo.sync_dropFK') IS NOT NULL
    DROP TABLE sync_dropFK
    
    CREATE TABLE sync_dropFK
        (
        ID INT IDENTITY (1,1) NOT NULL
        , Script NVARCHAR(4000)
        )
    
    DECLARE FKcursor CURSOR FOR
    
        SELECT 
            OBJECT_SCHEMA_NAME(parent_object_id)
            , OBJECT_NAME(parent_object_id)
            , name
        FROM 
            sys.foreign_keys WITH (NOLOCK)
        ORDER BY 
            1,2;
    
    OPEN FKcursor;
    
    FETCH NEXT FROM FKcursor INTO 
        @schema_name
        , @table_name
        , @constraint_name
    
    WHILE @@FETCH_STATUS = 0
    
    BEGIN
            SET @tsql = 'ALTER TABLE '
                    + QUOTENAME(@schema_name) 
                    + '.' 
                    + QUOTENAME(@table_name)
                    + ' DROP CONSTRAINT ' 
                    + QUOTENAME(@constraint_name) 
                    + ';';
        --PRINT @tsql;
        INSERT sync_dropFK  (
                            Script
                            )
                            VALUES (
                                    @tsql
                                    )   
    
        FETCH NEXT FROM FKcursor INTO 
        @schema_name
        , @table_name
        , @constraint_name
        ;
    
    END;
    
    CLOSE FKcursor;
    
    DEALLOCATE FKcursor;
    
    
    ---------------Generate scripts to create all existing foreign keys in a database --------------------------------
    ----------------------------------------------------------------------------------------------------------
    IF OBJECT_ID('dbo.sync_createFK') IS NOT NULL
    DROP TABLE sync_createFK
    
    CREATE TABLE sync_createFK
        (
        ID INT IDENTITY (1,1) NOT NULL
        , Script NVARCHAR(4000)
        )
    
    IF OBJECT_ID('dbo.sync_createCHECK') IS NOT NULL
    DROP TABLE sync_createCHECK
    
    CREATE TABLE sync_createCHECK
        (
        ID INT IDENTITY (1,1) NOT NULL
        , Script NVARCHAR(4000)
        )   
    
    DECLARE FKcursor CURSOR FOR
    
         SELECT 
            OBJECT_SCHEMA_NAME(parent_object_id)
            , OBJECT_NAME(parent_object_id)
            , name
            , OBJECT_NAME(referenced_object_id)
            , OBJECT_ID
            , is_disabled
            , is_not_for_replication
            , is_not_trusted
            , delete_referential_action
            , update_referential_action
            , OBJECT_SCHEMA_NAME(referenced_object_id)
    
        FROM 
            sys.foreign_keys WITH (NOLOCK)
    
        ORDER BY 
            1,2;
    
    OPEN FKcursor;
    
    FETCH NEXT FROM FKcursor INTO 
        @schema_name
        , @table_name
        , @constraint_name
        , @referenced_object_name
        , @constraint_object_id
        , @is_disabled
        , @is_not_for_replication
        , @is_not_trusted
        , @delete_referential_action
        , @update_referential_action
        , @referenced_schema_name;
    
    WHILE @@FETCH_STATUS = 0
    
    BEGIN
    
            BEGIN
                SET @tsql = 'ALTER TABLE '
                            + QUOTENAME(@schema_name) 
                            + '.' 
                            + QUOTENAME(@table_name)
                            +   CASE 
                                    @is_not_trusted
                                    WHEN 0 THEN ' WITH CHECK '
                                    ELSE ' WITH NOCHECK '
                                END
                            + ' ADD CONSTRAINT ' 
                            + QUOTENAME(@constraint_name)
                            + ' FOREIGN KEY (';
    
            SET @tsql2 = '';
    
            DECLARE ColumnCursor CURSOR FOR 
    
                SELECT 
                    COL_NAME(fk.parent_object_id
                    , fkc.parent_column_id)
                    , COL_NAME(fk.referenced_object_id
                    , fkc.referenced_column_id)
    
                FROM 
                    sys.foreign_keys fk WITH (NOLOCK)
                    INNER JOIN sys.foreign_key_columns fkc WITH (NOLOCK) ON fk.[object_id] = fkc.constraint_object_id
    
                WHERE 
                    fkc.constraint_object_id = @constraint_object_id
    
                ORDER BY 
                    fkc.constraint_column_id;
    
            OPEN ColumnCursor;
    
            SET @col1 = 1;
    
            FETCH NEXT FROM ColumnCursor INTO @fkCol, @pkCol;
    
            WHILE @@FETCH_STATUS = 0
    
            BEGIN
                IF (@col1 = 1)
                    SET @col1 = 0;
                ELSE
                BEGIN
                    SET @tsql = @tsql + ',';
                    SET @tsql2 = @tsql2 + ',';
                END;
    
                SET @tsql = @tsql + QUOTENAME(@fkCol);
                SET @tsql2 = @tsql2 + QUOTENAME(@pkCol);
                --PRINT '@tsql = ' + @tsql 
                --PRINT '@tsql2 = ' + @tsql2
                FETCH NEXT FROM ColumnCursor INTO @fkCol, @pkCol;
                --PRINT 'FK Column ' + @fkCol
                --PRINT 'PK Column ' + @pkCol 
            END;
    
            CLOSE ColumnCursor;
            DEALLOCATE ColumnCursor;
    
            SET @tsql = @tsql + ' ) REFERENCES ' 
                        + QUOTENAME(@referenced_schema_name) 
                        + '.' 
                        + QUOTENAME(@referenced_object_name)
                        + ' (' 
                        + @tsql2 + ')';
    
            SET @tsql = @tsql
                        + ' ON UPDATE ' 
                        + 
                            CASE @update_referential_action
                                WHEN 0 THEN 'NO ACTION '
                                WHEN 1 THEN 'CASCADE '
                                WHEN 2 THEN 'SET NULL '
                                    ELSE 'SET DEFAULT '
                            END
    
                        + ' ON DELETE ' 
                        + 
                            CASE @delete_referential_action
                                WHEN 0 THEN 'NO ACTION '
                                WHEN 1 THEN 'CASCADE '
                                WHEN 2 THEN 'SET NULL '
                                    ELSE 'SET DEFAULT '
                            END
    
                        + 
                        CASE @is_not_for_replication
                            WHEN 1 THEN ' NOT FOR REPLICATION '
                                ELSE ''
                        END
                        + ';';
    
            END;
    
        --  PRINT @tsql
            INSERT sync_createFK    
                            (
                            Script
                            )
                            VALUES (
                                    @tsql
                                    )
    
    -------------------Generate CHECK CONSTRAINT scripts for a database ------------------------------
    ----------------------------------------------------------------------------------------------------------
    
            BEGIN
    
            SET @tsql = 'ALTER TABLE '
                        + QUOTENAME(@schema_name) 
                        + '.' 
                        + QUOTENAME(@table_name)
                        + 
                            CASE @is_disabled
                                WHEN 0 THEN ' CHECK '
                                    ELSE ' NOCHECK '
                            END
                        + 'CONSTRAINT ' 
                        + QUOTENAME(@constraint_name)
                        + ';';
            --PRINT @tsql;
            INSERT sync_createCHECK 
                            (
                            Script
                            )
                            VALUES (
                                    @tsql
                                    )   
            END;
    
        FETCH NEXT FROM FKcursor INTO 
        @schema_name
        , @table_name
        , @constraint_name
        , @referenced_object_name
        , @constraint_object_id
        , @is_disabled
        , @is_not_for_replication
        , @is_not_trusted
        , @delete_referential_action
        , @update_referential_action
        , @referenced_schema_name;
    
    END;
    
    CLOSE FKcursor;
    
    DEALLOCATE FKcursor;
    
    --SELECT * FROM sync_DropFK
    --SELECT * FROM sync_CreateFK
    --SELECT * FROM sync_CreateCHECK
    ---------------------------------------------------------------------------
    2.)
    -----------------------------------------------------------------------------------------------------------------
    ----------------------------execute Drop FK Scripts --------------------------------------------------
    
    DECLARE @scriptD NVARCHAR(4000)
    
    DECLARE DropFKCursor CURSOR FOR
        SELECT Script 
        FROM sync_dropFK WITH (NOLOCK)
    
    OPEN DropFKCursor
    
    FETCH NEXT FROM DropFKCursor
    INTO @scriptD
    
    WHILE @@FETCH_STATUS = 0
    BEGIN
    --PRINT @scriptD
    EXEC (@scriptD)
    FETCH NEXT FROM DropFKCursor
    INTO @scriptD
    END
    CLOSE DropFKCursor
    DEALLOCATE DropFKCursor
    --------------------------------------------------------------------------------
    3.) 
    
    ------------------------------------------------------------------------------------------------------------------
    ----------------------------Truncate all tables in the database other than our staging tables --------------------
    ------------------------------------------------------------------------------------------------------------------
    
    
    EXEC sp_MSforeachtable 'IF OBJECT_ID(''?'') NOT IN 
    (
    ISNULL(OBJECT_ID(''dbo.sync_createCHECK''),0),
    ISNULL(OBJECT_ID(''dbo.sync_createFK''),0),
    ISNULL(OBJECT_ID(''dbo.sync_dropFK''),0)
    )
    BEGIN TRY
     TRUNCATE TABLE ?
    END TRY
    BEGIN CATCH
     PRINT ''Truncation failed on''+ ? +''
    END CATCH;' 
    GO
    -------------------------------------------------------------------------------
    -------------------------------------------------------------------------------------------------
    ----------------------------execute Create FK Scripts and CHECK CONSTRAINT Scripts---------------
    ----------------------------tack me at the end of the ETL in a SQL task-------------------------
    -------------------------------------------------------------------------------------------------
    DECLARE @scriptC NVARCHAR(4000)
    
    DECLARE CreateFKCursor CURSOR FOR
        SELECT Script 
        FROM sync_createFK WITH (NOLOCK)
    
    OPEN CreateFKCursor
    
    FETCH NEXT FROM CreateFKCursor
    INTO @scriptC
    
    WHILE @@FETCH_STATUS = 0
    BEGIN
    --PRINT @scriptC
    EXEC (@scriptC)
    FETCH NEXT FROM CreateFKCursor
    INTO @scriptC
    END
    CLOSE CreateFKCursor
    DEALLOCATE CreateFKCursor
    -------------------------------------------------------------------------------------------------
    DECLARE @scriptCh NVARCHAR(4000)
    
    DECLARE CreateCHECKCursor CURSOR FOR
        SELECT Script 
        FROM sync_createCHECK WITH (NOLOCK)
    
    OPEN CreateCHECKCursor
    
    FETCH NEXT FROM CreateCHECKCursor
    INTO @scriptCh
    
    WHILE @@FETCH_STATUS = 0
    BEGIN
    --PRINT @scriptCh
    EXEC (@scriptCh)
    FETCH NEXT FROM CreateCHECKCursor
    INTO @scriptCh
    END
    CLOSE CreateCHECKCursor
    DEALLOCATE CreateCHECKCursor
    

提交回复
热议问题