• [翻译]SQL Server 未公开的两个存储过程sp_MSforeachtable 和 sp_MSforeachdb


    SQL Server 未公开的两个存储过程sp_MSforeachtable 和 sp_MSforeachdb

    您是否曾经写过代码来处理数据库中的所有表?处理一个 SQL Server实例中的所有数据库的代码又该如何写?然则,您是否知道有多种方法可以解决这问题?您可以创建一个游标cursor包含所有数据表,或者包含SQL Server实例的所有数据库;或者使用非公开(undocumented)的存储过程。本文将向您阐述非公开的存储过程的工作方式,以及应用实例向您展示如何使用它们。非公开的存储过程比之游标更易用。

    概述

    Microsoft 提供了两个非公开化的存储过程,让您可以迭代处理数据库中的所有表,或者SQL Server 实例中的所有数据库。第一个存储过程是"sp_MSforeachtable",让您可以轻易地使用代码处理数据库中的所有表;另一个是"sp_MSforeachdb",处理SQL Server 实例中的所有数据库。让我们深入地了解这两个存储过程。

    sp_MSforeachtable

    "sp_MSforeachtable"没有在在线文档中出现,它存在于master数据库中,可以对给定数据库的所有表执行单条或多条T-SQL命令,请看下面的例子。

    假如,您需要创建一个临时表,记录当前数据库拥有的表的表名、行记录数。为了实现此功能,您需要执行这样的命令:"select '<mytable>', count(*) from <mytable>"。其中"<mytable>"替换为数据库中的每个表名,并将结果插入到临时表。下面我们用游标与非公开的"sp_MSforeachtable"来分别实现。

    使用游标的方式:

    下面是输出结果:

    下面代码应用非公开的"sp_MSforeachtable"生成相同的结果:

    下面是结果:

    可见,使用游标与sp_MSforeachtable可生成相同的结果,您认为哪种方式更具可读性,更简单?下面来详细介绍sp_MSforeachtable的语法:

    exec @RETURN_VALUE=sp_MSforeachtable @command1, @replacechar, @command2,

    @command3, @whereand, @precommand, @postcommand

    说明:

    • @RETURN_VALUE – 返回值
    • @command1 – 类型是nvarchar(2000),sp_MSforeachtable最先执行的命令
    • @replacechar – 处理过程中,将命令行的这个字符替换为具体的表名(默认是"?")
    • @command2\@command3:对每个数据表,都会执行这两条命令,@command2在@command1之后执行,@command3在@command2之后执行
    • @whereand – 类型是varchar(2000),提供额外的约束来过滤 sysobjects 表的行
    • @precommand - 类型是varchar(2000),在处理任何表之前执行此命令
    • @postcommand - 类型是varchar(2000),在处理完所有表之后执行此命令

    下面几个例子演示此存储过程的用法,处理所有表或者部分表。

    下面查询以字母 p 开头的表,使用参数 @whereand 设置过滤条件,代码如下:

    下面是结果:

    上面的代码使用了参数 @command1 与 @whereand,参数 @whereand 用来设置 WHERE 条件,筛选出以字母 p 开头的表名,我设置了参数值为"and o.name like ''p%''"。如果您希望使用多个条件约束,如以 p 开头或者以 a 开头,设置参数值为:

    and o.name like ''p%'' or o.name like ''a%''

     

    如果语句有问题,将 name 的前缀去掉,如下:

     

    and name like ''p%'' or name like ''a%''

     

    注意,上面例子的参数 @command1 使用了"?",它叫做替换字符(replacement character),默认被所有表名替换。如果您需要在命令中使用"?"作为内容而不是被表名替换的替换字符,那么可以使用参数 @replacechar 来设置替换字符。下面例子使用"{"作为替换字符。

    下面是结果:

    还有两个参数 @precommand 与 @postcommand,看下面例子,把上面例子中的所有语句整合为一个简洁的存储过程调用。

    注意上面例子用了全局临时表 ##rowcount,如果用临时表 #rowcount会报错。参数 @precommand 创建全局临时表,只执行了一次,并先于 @command1 的语句执行。@postcommmand 的语句待迭代处理完所有表后执行,也仅执行一次,用于显示结果并删除临时表。

     

    sp_MSforeachdb

    sp_MSforeachdb 同样也是在 master 数据库中,它迭代SQL Server 实例中的每个数据库,以执行T-SQL 语句,如"DBCCCHECKDB",在看看它的语法

    exec @RETURN_VALUE = sp_MSforeachdb @command1, @replacechar,

    @command2, @command3, @precommand, @postcommand

    说明:

    • @RETURN_VALUE – 返回值
    • @command1 – 类型是 nvarchar(2000),最先执行的命令
    • @replacechar – 替换字符,命令字符串中被替换为实际的数据库名(默认是"?")
    • @command2\@command3:对每个数据库,都会执行这两条命令,@command2在@command1之后执行,@command3在@command2之后执行
    • @precommand - 类型是varchar(2000),在处理任何数据库之前执行此命令
    • @postcommand - 类型是varchar(2000),在处理完所有数据库之后执行此命令

    sp_MSforeachdb 的参数与sp_MSforeachtable 的参数类似,因此,不再特意介绍这些参数。

    请看下面的简单例子,此例子将进行数据库备份,然后对每个数据库做"DBCC CHECKDB":

    这里我用了三条不同的命令,第一条打印正在处理的数据库名。sp_MSforeachtable 有一个参数用来过滤需要处理的数据表,但是sp_MSforeachdb没有类似的过滤参数。由于SQL Server 不支持对 tempdb 的备份,因此我要跳过tempdb,这是我在每条命令使用 IF 的原因。第二条命令进行数据库备份,最后一条命令对除 tempdb 之外的数据库运行"DBCC CHECKDB"。

    运行上面命令之前要先创建目录"c:\temp",下面是部分输出结果:

     

    使用SQL Server非公开存储过程的说明

    当使用这些非公开的存储过程时您须小心,并进行测试。由于未公开,意味着Microsoft在任何版本的升级或者补丁包都可能对它们进行修改,并且不做任何告知。因此,您需要在所有的SQL Server版本做全面的测试,测试以验证您的代码是否在新版本中仍然正常运行。

    结语

    正如您所见,这两个非公开的存储过程比游标易用,以后您可以用它们来迭代处理数据表或数据库。但是请谨记,这两个存储过程是非公开的,Microsoft很可能会随时改变它们的功能。

    参考

    SQL Server Undocumented Stored Procedures sp_MSforeachtable and sp_MSforeachdb

    sp_MSforeachtable

     

  • 相关阅读:
    Oracle instr() 字符查找函数
    Oracle 中触发器增加存储过程commit问题
    Oracle 记录下jdbc thin client module名称
    sqoop job 实现自动增量导入
    Linux LVM--三种Logic Volume
    Kafka ISR and AR HW 、 LEO
    Kafka Rebalance机制分析
    Kafka 基础操作
    Kafka 通过python简单的生产消费实现
    Kafka为什么速度那么快?该怎么回答
  • 原文地址:https://www.cnblogs.com/feixian49/p/2042733.html
Copyright © 2020-2023  润新知