循环遍历服务器以获取统计信息 (SSMS) 时找不到存储过程 'sp_msforeachtable'
Could not find stored procedure 'sp_msforeachtable' while looping through server for stats (SSMS)
我写这个是为了遍历服务器上的每个数据库,收集每个 table 的统计信息并将它们存储在临时 table 中。最终,我会将它整合到一个更永久的结构中,但现在我只是想让它发挥作用。我的问题是,在 57 个数据库之后,我收到错误消息,指出找不到存储过程 sp_msforeachtable.
我已验证此存储过程存在于服务器上和服务器级别的每个数据库中。
我通过将其添加到“where name not in”条件中将其排除在调查结果中,它只是移动到列表中的下一个并给出相同的错误。(我已经确认也存在于下一个数据库中)。我实际上已经为接下来的 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
既然您已经验证存储过程确实存在,我相信您的数据库是区分大小写的。因此,错误仍然准确。基本上,您使用的案例的存储过程不存在。实际过程名称是 sp_MSforeachtable
在您的代码中,您使用了以下内容:
执行 sp_msforeachtable
如果您更改代码以使用存储过程的正确大小写为 sp_MSforeachtable,它应该可以工作:
SET @Sql1 =
'USE ' + @DbName + '; ' + '
Exec sp_MSforeachtable
''insert into #TableLvlSizes exec sp_spaceused [?]'''
如果有人在使用 Azure SQL,他们将找不到 sp_MSforeachtable
,因为它在 Azure SQL 中不可用。
您可能需要为自己创建一个。
我写这个是为了遍历服务器上的每个数据库,收集每个 table 的统计信息并将它们存储在临时 table 中。最终,我会将它整合到一个更永久的结构中,但现在我只是想让它发挥作用。我的问题是,在 57 个数据库之后,我收到错误消息,指出找不到存储过程 sp_msforeachtable.
我已验证此存储过程存在于服务器上和服务器级别的每个数据库中。
我通过将其添加到“where name not in”条件中将其排除在调查结果中,它只是移动到列表中的下一个并给出相同的错误。(我已经确认也存在于下一个数据库中)。我实际上已经为接下来的 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
既然您已经验证存储过程确实存在,我相信您的数据库是区分大小写的。因此,错误仍然准确。基本上,您使用的案例的存储过程不存在。实际过程名称是 sp_MSforeachtable
在您的代码中,您使用了以下内容: 执行 sp_msforeachtable
如果您更改代码以使用存储过程的正确大小写为 sp_MSforeachtable,它应该可以工作:
SET @Sql1 =
'USE ' + @DbName + '; ' + '
Exec sp_MSforeachtable
''insert into #TableLvlSizes exec sp_spaceused [?]'''
如果有人在使用 Azure SQL,他们将找不到 sp_MSforeachtable
,因为它在 Azure SQL 中不可用。
您可能需要为自己创建一个。