Categories: SQL Server

MSSQL Check Index Fragmentation

Query to find the fragmentation status of all indexes on a single database. Line 11 you need to insert your db name.

SELECT ps.database_id, ps.OBJECT_ID,
 FROM sys.dm_db_index_physical_stats (DB_ID(), NULL, NULL, NULL, NULL) AS ps
 INNER JOIN sys.indexes AS b ON ps.OBJECT_ID = b.OBJECT_ID
 AND ps.index_id = b.index_id
 WHERE ps.database_id = DB_ID()
 FROM sys.dm_db_index_physical_stats(DB_ID('DatabaseName'), NULL, NULL, NULL , NULL);
 -- Ensure a USE <databasename> statement has been executed first.
 DECLARE @objectid int;
 DECLARE @indexid int;
 DECLARE @partitioncount bigint;
 DECLARE @schemaname nvarchar(130);
 DECLARE @objectname nvarchar(130);
 DECLARE @indexname nvarchar(130);
 DECLARE @partitionnum bigint;
 DECLARE @partitions bigint;
 DECLARE @frag float;
 DECLARE @command nvarchar(4000);
 -- Conditionally select tables and indexes from the sys.dm_db_index_physical_stats function
 -- and convert object and index IDs to names.
 object_id AS objectid,
 index_id AS indexid,
 partition_number AS partitionnum,
 avg_fragmentation_in_percent AS frag
 INTO #work_to_do
 FROM sys.dm_db_index_physical_stats (DB_ID(), NULL, NULL , NULL, 'LIMITED')
 WHERE avg_fragmentation_in_percent > 10.0 AND index_id > 0;
 -- Declare the cursor for the list of partitions to be processed.
 DECLARE partitions CURSOR FOR SELECT * FROM #work_to_do;
 -- Open the cursor.
 OPEN partitions;
 -- Loop through the partitions.
 WHILE (1=1)
 FROM partitions
 INTO @objectid, @indexid, @partitionnum, @frag;
 SELECT @objectname = QUOTENAME(, @schemaname = QUOTENAME(
 FROM sys.objects AS o
 JOIN sys.schemas as s ON s.schema_id = o.schema_id
 WHERE o.object_id = @objectid;
 SELECT @indexname = QUOTENAME(name)
 FROM sys.indexes
 WHERE object_id = @objectid AND index_id = @indexid;
 SELECT @partitioncount = count (*)
 FROM sys.partitions
 WHERE object_id = @objectid AND index_id = @indexid;
 -- 30 is an arbitrary decision point at which to switch between reorganizing and rebuilding.
 IF @frag < 30.0
 SET @command = N'ALTER INDEX ' + @indexname + N' ON ' + @schemaname + N'.' + @objectname + N' REORGANIZE';
 IF @frag >= 30.0
 SET @command = N'ALTER INDEX ' + @indexname + N' ON ' + @schemaname + N'.' + @objectname + N' REBUILD';
 IF @partitioncount > 1
 SET @command = @command + N' PARTITION=' + CAST(@partitionnum AS nvarchar(10));
 EXEC (@command);
 PRINT N'Executed: ' + @command;
 -- Close and deallocate the cursor.
 CLOSE partitions;
 DEALLOCATE partitions;
 -- Drop the temporary table.
 DROP TABLE #work_to_do;


Disqus Comments Loading...
Published by

Recent Posts

Bittorrent IP Blocklists

In addition to using a VPN service, as an extra precaution I've been using the blocklist feature of my bittorrent… Read More

October 26, 2019 3:31 pm

FreeNAS Error Creating Pool

command '('gpart', 'create', '-s', 'gpt', '/dev/da8')' returned non-zero exit status 1. If you get this error while trying to create… Read More

June 7, 2019 3:44 pm

Change Grub Default Boot Entry on Linux Mint

I'm dual booting Windows and Linux Mint on my laptop. The grub default is to boot into Linux Mint, however… Read More

April 23, 2019 7:45 pm

How to Reset Secure Channel On Active Directory Domain Controller

When you're a little too careless about virtualizing your domain controllers, cloning, migrating, backing up and restoring, returning from vacation… Read More

April 21, 2019 8:14 am

Run Systemd Script Before System Shutdown

I tried to retain the NGINX FastCGI cache and have it persist across system reboots instead of being ephemeral by… Read More

April 20, 2019 10:14 am

Learn Systemctl Usage to Manage Systemd Service in Linux

Systemd is new service manager for Linux. It's a replacement for all previous init systems (SysV/SysVinit & Ubuntu's Upstart) and… Read More

April 20, 2019 7:55 am