1

我有大量的存储过程,它们从我们正在处理的各种环境中复制一个实体(跨越 30 多个表):Dev/Integration/PreProd/Prod

这个想法有时是数据库结构从一个平台更改为另一个。一般来说,我们会进行一些小的更改,例如将 1-2 列移动到新表或将新表添加到结构等。

考虑到这种复制的复杂性,我希望在所有环境中都拥有这些程序的单一版本,无论与数据库结构相关的差异如何,以便可以从任何环境复制到另一个环境。例如,如果该列存在于srcdest => 中,则执行该逻辑,否则执行其他逻辑等。

这个想法是我不能在存储过程中使用不存在的列,即使代码检查列的存在并且永远不会执行该代码分支。

最好的选择是使用 编写代码sp_executesql,但代码变得非常复杂:考虑到一旦结构更改在所有环境中对齐,我将不得不丢弃部分代码,这太过分了。

4

1 回答 1

2

我会确保您需要的所有表和所有未来的表都具有可识别的命名约定,以便您可以使用 sys.all_objects 和 sys.all_columns 表查询要复制的表/列。在此示例中,您可以通过 @SourceDB 参数将源数据库名称传递给存储过程,并在源/目标表/列名称相交的位置执行插入。未指定目标数据库,因为它被假定为当前数据库:

-- get intersecting table/column names from @SourceDB
-- and Destination (Current) DB
declare @sql nvarchar(max)
set @sql = 'select o.name,c.name
    from ' + @SourceDB + '.sys.all_objects o
        inner join ' + @SourceDB + '.sys.all_columns c
            on c.object_id = o.object_id
    where o.type = ''U''
    and o.name like ''SomeStringToIdentifyYourTables%''
    intersect
    select o.name,c.name
    from sys.all_objects o
        inner join sys.all_columns c
            on c.object_id = o.object_id
    where o.type = ''U''
    and o.name like ''SomeStringToIdentifyYourTables%'''   
declare @TblColNames table(TblName nvarchar(200),ColName nvarchar(200))
insert into @TblColNames
exec(@sql)

-- insert data from Source DB to Destination (Current) DB
-- where Table/Column Names Intersect
declare @sql nvarchar(max)
set @sql = ''
select @sql = @sql + 'insert into ' +
    o.name + ' (' + replace((
                    select c.name +
                        case row_number() over(order by c.name desc)
                        when 1 then '' else '||' end
                    from @TblColNames c
                    where c.TblName = o.TblName
                    order by c.name asc
                    for xml path('')),'||',',') + ')
    select ' + replace((
                    select c.name +
                        case row_number() over(order by c.name desc)
                        when 1 then '' else '||' end
                    from @TblColNames c
                    where c.TblName = o.TblName
                    order by c.name asc
                    for xml path('')),'||',',') + '
    from ' + @SourceDB + '..' + o.name + '
'
from @TblColNames o
exec(@sql)
于 2014-04-01T00:21:38.087 回答