循环通过服务器获取统计信息时找不到存储过程“sp_msforeachtable”(SSMS)

Joe*_*e G 5 sql-server stored-procedures ssms-2016

我编写此代码是为了循环访问服务器上的每个数据库,收集每个表的统计信息并将它们存储在临时表中。最终,我会将其集成到一个更永久的结构中,但现在我只是想让它发挥作用。我的问题是,在 57 个数据库之后,我收到错误消息,指出找不到存储过程 sp_msforeachtable。

我已经验证该存储过程存在于服务器上和服务器级别的每个数据库上。

我通过将其添加到“名称不在”条件中,在结果中排​​除了该数据库,它只是移动到列表中的下一个数据库并给出相同的错误。(我已确认它存在于下一个数据库中还)。实际上我已经为接下来的 6 个数据库做到了这一点。

这导致我无法收集准确的信息。我是否在某个地方耗尽了资源?

                    DECLARE @Database TABLE (DbName SYSNAME);
                    
                    IF OBJECT_ID('tempdb.dbo.#TableLvlSizes', 'U') IS NOT NULL 
                    BEGIN
                           PRINT 'dropping table'
                        DROP TABLE tempdb.dbo.#TableLvlSizes;
                    END    
                    
                    CREATE TABLE #TableLvlSizes (
                        TableName nvarchar(128)
                        ,NumberOfRows varchar(50)
                        ,ReservedSpace varchar(50)
                        ,TableDataSpace varchar(50)
                        ,IndexSize varchar(50)
                        ,unused varchar(50))
                    
                    DECLARE @DbName AS SYSNAME;
                    DECLARE @Sql1 AS VARCHAR(MAX);
                    
                    SET @DbName = '';
                    
                    INSERT INTO @Database (DbName)
                    SELECT NAME
                    FROM sys.databases
                    where name not in ('tempdb')
                    ORDER BY NAME ASC;
                    
                    WHILE @DbName IS NOT NULL
                    BEGIN
                        SET @DbName = (
                                SELECT MIN(DbName)
                                FROM @Database
                                WHERE DbName > @DbName
                                );
                            print @DbName;
                            SET @Sql1 =
                        'USE ' + @DbName + '; ' + '
                            Exec sp_msforeachtable
                            ''insert into #TableLvlSizes exec sp_spaceused [?]''
                        '
                        Exec (@SQL1);
                    END
Run Code Online (Sandbox Code Playgroud)

Pal*_*ine 15

如果有人使用 Azure SQL,他们将找不到,sp_MSforeachtable因为它在 Azure SQL 中不可用。

您可能需要为自己创建一个。


小智 5

由于您已经验证存储过程确实存在,因此我相信您的数据库是区分大小写的。因此,该误差仍然是准确的。基本上,您使用的情况的存储过程不存在。实际的过程名称是 sp_ MS foreachtable

在您的代码中,您使用以下命令:Exec sp_ ms foreachtable

如果您更改代码以使用正确的存储过程大小写为 sp_ MS foreachtable,它应该可以工作:

SET @Sql1 =
            'USE ' + @DbName + '; ' + '
             Exec sp_MSforeachtable
             ''insert into #TableLvlSizes exec sp_spaceused [?]'''
Run Code Online (Sandbox Code Playgroud)