Измените хранимую процедуру Nested Cursor на Nested loop - PullRequest
0 голосов
/ 14 мая 2019

У меня есть хранимая процедура, которая в настоящее время использует вложенные курсоры для создания представлений, используя параметры базы данных, таблицы, столбца. Мне нужно, чтобы это было изменено на запрос цикла while. Я понимаю, что результаты одинаковы, но мы просто предпочитаем не использовать курсоры.

Это кажется очень сложным из-за динамического SQL.

ALTER PROCEDURE [dbo].[spReCreateCommonViews_AK] 
AS
    DECLARE @Exec INT = 1, @Print INT = 2
    -- DECLARE @debug INT = @Exec
    DECLARE @debug INT = @Print

    DECLARE @Sql VARCHAR(MAX),
            @SourceName VARCHAR(100),
            @DatabaseName VARCHAR(100),
            @CompanyName VARCHAR(100),
            @ColumnCursor VARCHAR(800),
            @TableId INT,
            @TableName VARCHAR(100),
            @ColumnType INT,
            @ColumnName VARCHAR(100),
            @ColumnList VARCHAR(MAX) = '',
            @CreateViewSQL VARCHAR(MAX)

    /**  COMPANIES ********/
    DECLARE CompaniesCursor CURSOR FOR
        SELECT
            [Database],
            REPLACE(REPLACE(REPLACE([Name], '.', '_'), '/', '_'), '''', '_') AS [Name],
            CASE 
               WHEN RIGHT([Source], 7) = '_SOURCE'  
                  THEN '[' + [Source] + '].[' + [Database] + ']' 
                  ELSE '[' + [Database] + ']' 
            END AS [Source]
        FROM
            Company 

    OPEN CompaniesCursor
    FETCH NEXT FROM CompaniesCursor INTO @DatabaseName, @CompanyName, @SourceName

    WHILE @@fetch_status = 0
    BEGIN
        /**  TABLES  ***********/
        SELECT @Sql = 'DECLARE TablesCursor CURSOR FOR 
                                        select 
                                            o.[id] as TableId,
                                            o.[name] as TableName
                                        from '+@SourceName+'.dbo.sysobjects o 
                                        inner join '+@SourceName+'.dbo.[Object] t 
                                        on o.[name] = replace(replace(replace(t.[Name],''.'',''_''),''/'',''_''),'''''''',''_'')
                                        and t.[Company Name] = ''''
                                        and t.[Type] = 0
                                        where o.[name] <> ''Company''
                                        and o.[name] not like ''' + @CompanyName + '$%''
                                            and o.type = ''U'' 
                                        order by o.[name]

                                '

                                    exec(@Sql)
                                    open TablesCursor
                                    fetch next from TablesCursor into @TableId, @TableName
                                    while @@fetch_status = 0
                                    begin


                                                                                    /**  COLUMNS  ************************************************************/
                                                                                    select @ColumnCursor = 
                                                                                    '
                                                                                    declare ColumnsCursor cursor
                                                                                    for 
                                                                                        select ''['' + [name] + '']'' as ColumnName, xtype as ColumnType
                                                                                        from '+@SourceName+'.dbo.syscolumns 
                                                                                        where [id] = ' + convert(varchar, @TableId) + '
                                                                                        order by colid
                                                                                    '
                                                                                    exec(@ColumnCursor)
                                                                                    open ColumnsCursor
                                                                                    fetch next from ColumnsCursor into @ColumnName, @ColumnType
                                                                                    while @@fetch_status = 0
                                                                                    begin

                                                                                                                                    select @ColumnList = @ColumnList + @ColumnName + case when @ColumnType in (35, 99, 167, 175, 231, 239) then ' collate database_default as ' + @ColumnName + ', ' else ', ' end


                                                                                    -----------------------------
                                                                                    fetch next from ColumnsCursor into @ColumnName, @ColumnType
                                                                                    end
                                                                                    close ColumnsCursor
                                                                                    deallocate ColumnsCursor
                                                                                    -----------------------------




                                select @CreateViewSQL= 'if object_id(N'''+@CompanyName+'$' + @TableName + ''', N''V'') is not null drop view ['+@CompanyName+'$' + @TableName + ']'
                                if @debug & @Print = @Print print(@CreateViewSQL)
                                if @debug & @Exec = @Exec exec (@CreateViewSQL)

                                select @CreateViewSQL= 'create view ['+@CompanyName+'$' + @TableName + '] as select ' + left(@ColumnList,len(@ColumnList)-1) + ' from ' + @SourceName + '.dbo.[' + @TableName + ']'
                                if @debug & @Print = @Print print(@CreateViewSQL)
                                if @debug & @Exec = @Exec exec (@CreateViewSQL)

                                select @ColumnList = ''


                                -----------------------------
                                fetch next from TablesCursor into @TableId, @TableName
                                end
                                close TablesCursor
                                deallocate TablesCursor
                                ------------------------------          

                select @Sql = '
                        insert into [Object] (
                        [Type],
                        [Company Name],
                        [ID],
                        [Name],
                        [Modified],
                        [Compiled],
                        [BLOB Reference],
                        [BLOB Size],
                        [DBM Table No_],
                        [Date],
                        [Time],
                        [Version List]
                    )
                    select 
                        o.[Type],
                        '''+@CompanyName+''' as [Company Name],
                        o.[ID],
                        o.[Name] collate database_default,
                        o.[Modified],
                        o.[Compiled],
                        o.[BLOB Reference],
                        o.[BLOB Size],
                        o.[DBM Table No_],
                        o.[Date],
                        o.[Time],
                        o.[Version List] collate database_default 
                    from '+@SourceName+'.dbo.Object o 
                    where o.[Name] <> ''Company''
                    and o.[Company Name] = ''''
                    and o.[Type] = 0
                    and not exists (
                        select c.*
                        from [Object] c
                        where c.[Name] = o.[Name] collate database_default 
                        and c.[Company Name] = '''+@CompanyName+'''
                        and c.[Type] = 0
                        and c.[ID] = o.[ID]
                    )
                '

                if @debug & @Print = @Print
                    print(@Sql)
                if @debug & @Exec = @Exec
                    exec (@Sql)

                -----------------------------
                fetch next from CompaniesCursor into @DatabaseName, @CompanyName, @SourceName
                end
                close CompaniesCursor
                deallocate CompaniesCursor
                -----------------------------
...