36

我正在运行 DTS 以在我的数据库中执行任务,首先我需要禁用数据库中的所有索引并在 DTS 完成工作时重新启用它们。

有没有办法可以禁用整个数据库中的所有索引,然后重新启用它们?

我知道如何一个一个地禁用/启用,有人可以帮助我一次性禁用/启用所有的方法,作为 DTS 中的一个步骤。

4

7 回答 7

50

我们可以使用下面的脚本来禁用索引

ALTER INDEX ALL ON [TableName]
DISABLE;

将您的批量插入到表中,然后在脚本下面运行。

ALTER INDEX ALL ON [TableName]
REBUILD;
于 2016-02-18T13:01:55.633 回答
44

这是一个脚本,它将为数据库中的所有非聚集索引输出 ALTER 语句。您可以轻松修改它以输出 REBUILD 脚本和聚集索引的脚本

select 'ALTER INDEX [' + I.name + '] ON [' + T.name + '] DISABLE' 
from sys.indexes I
inner join sys.tables T on I.object_id = T.object_id
where I.type_desc = 'NONCLUSTERED'
and I.name is not null
于 2013-08-15T10:33:09.337 回答
29

这适用于 SQL Server 2008 和更新版本。它允许不同的模式以及包含空格、破折号和其他必须引用的特殊字符的名称。sql语句中方括号[]有什么用?

这些脚本会将代码输出到结果选项卡中。您必须复制/粘贴到查询选项卡中并执行它们。

禁用脚本

SELECT 'ALTER INDEX ' + QUOTENAME(I.name) + ' ON ' +  QUOTENAME(SCHEMA_NAME(T.schema_id))+'.'+ QUOTENAME(T.name) + ' DISABLE' 
FROM sys.indexes I
INNER JOIN sys.tables T ON I.object_id = T.object_id
WHERE I.type_desc = 'NONCLUSTERED'
AND I.name IS NOT NULL
AND I.is_disabled = 0

启用脚本(重建)

SELECT 'ALTER INDEX ' + QUOTENAME(I.name) + ' ON ' +  QUOTENAME(SCHEMA_NAME(T.schema_id))+'.'+ QUOTENAME(T.name) + ' REBUILD' 
FROM sys.indexes I
INNER JOIN sys.tables T ON I.object_id = T.object_id
WHERE I.type_desc = 'NONCLUSTERED'
AND I.name IS NOT NULL
AND I.is_disabled = 1

这是基于此处的另一个答案。

于 2015-08-14T16:29:01.487 回答
5

为了启用索引,您必须重建它。此脚本将重建所有禁用的索引。

DECLARE @my_sql2 NVARCHAR(200);

DECLARE cur_rebuild CURSOR FOR 
   SELECT 'ALTER INDEX ' +  i.name + ' ON ' + t.name + ' REBUILD' FROM sys.indexes i JOIN sys.tables t ON i.object_id = t.object_id WHERE i.is_disabled = 1 ORDER BY t.name, i.name;
OPEN cur_rebuild;
FETCH NEXT FROM cur_rebuild INTO @my_sql2;
WHILE @@FETCH_STATUS = 0
   BEGIN
      EXECUTE sp_executesql  @my_sql2;
      FETCH NEXT FROM cur_rebuild INTO @my_sql2;
   END;
CLOSE cur_rebuild;
DEALLOCATE cur_rebuild;
GO
于 2014-12-03T05:02:58.167 回答
3

在加载大量数据时禁用索引是一个好主意,但是……最大的问题是聚集索引。如果禁用聚集索引,则禁用了整个表。

有几个选项建议自己,而且都不是简单的。

1)循环遍历系统视图(sys.indexes),提取表和索引名,生成并执行动态SQL来禁用索引。有一个“撤消”例程来重新启用它们。(小心——它是唯一索引还是唯一约束?)唉,这仅在您不使用聚集索引时才有效。祝你好运。

2) 至于 1,但跳过任何聚集索引。加载数据时,请确保它按(聚集索引)顺序加载,否则加载时间会很短,并且会出现碎片表。(如果你的数据提供者和我一样,那也祝你好运。)

3) 在数据库中创建包含“加载”表上的索引定义的表。构建一个循环遍历它们并删除所有索引(最后是聚集索引)的例程。如果您先截断表格,这将很快。加载您的数据,然后循环遍历并从头开始重新创建索引(首先集群)。使用表分区来减少系统其余部分的可怕性(例如,在“加载”表上执行上述所有操作,然后使用分区切换将加载的数据移动到“活动”表中)。我花了很多时间来构建这样一个系统,但它可以而且会起作用。

于 2013-08-14T15:56:58.487 回答
3

使用此脚本禁用所有索引

-- Disable All Indices
DECLARE @Script NVARCHAR(MAX)
DECLARE curIndices CURSOR FAST_FORWARD READ_ONLY FOR
    SELECT 'ALTER INDEX ' + QUOTENAME(indices.name) + ' ON ' + QUOTENAME(SCHEMA_NAME(tableNames.schema_id))+'.'+ QUOTENAME(tableNames.name) + ' DISABLE'
    FROM
        sys.indexes indices  INNER JOIN 
        sys.tables tableNames ON indices.object_id = tableNames.object_id
    WHERE 
        indices.type_desc = 'NONCLUSTERED' AND 
        indices.name IS NOT NULL AND
        indices.is_disabled = 0;
OPEN curIndices
FETCH NEXT FROM curIndices INTO @Script
WHILE @@FETCH_STATUS = 0
BEGIN
    PRINT @Script
    EXECUTE sp_executesql @Script 

    FETCH NEXT FROM curIndices INTO @Script
END
CLOSE curIndices
DEALLOCATE curIndices

使用此脚本重建(启用)所有索引

-- Rebuild All Indices
DECLARE @Script NVARCHAR(MAX)
DECLARE curIndices CURSOR FAST_FORWARD READ_ONLY FOR
    SELECT 'ALTER INDEX ' + QUOTENAME(indices.name) + ' ON ' + QUOTENAME(SCHEMA_NAME(tableNames.schema_id))+'.'+ QUOTENAME(tableNames.name) + ' REBUILD'
    FROM
        sys.indexes indices  INNER JOIN 
        sys.tables tableNames ON indices.object_id = tableNames.object_id
    WHERE 
        indices.type_desc = 'NONCLUSTERED' AND 
        indices.name IS NOT NULL AND
        indices.is_disabled = 1;
OPEN curIndices
FETCH NEXT FROM curIndices INTO @Script
WHILE @@FETCH_STATUS = 0
BEGIN
    PRINT @Script
    EXECUTE sp_executesql @Script 

    FETCH NEXT FROM curIndices INTO @Script
END
CLOSE curIndices
DEALLOCATE curIndices
于 2019-01-28T10:33:20.937 回答
2

您必须运行一个脚本来选择表和索引的元数据。然后你可以做一个:

ALTER INDEX indexname ON tablename DISABLE; 

稍后您可以运行类似的脚本来重建:

ALTER INDEX indexname ON tablename REBUILD; 

您可以一次执行这些,或者将它们收集到 NVARCHAR(MAX) 变量中并将它们作为单个批处理执行。您可以在前面的问题中看到示例代码:

禁用所有非聚集索引

于 2013-08-14T15:47:22.660 回答