Sql-server – SQL Server clustered index high fragmentation

clustered-indexfragmentationindexsql serversql-server-2012

I've got a Table with an Integer (4bytes) as primary key. it's defined as an identity. it is also the clustered index.

Inserts are working perfectly fine. After inserting 2000 rows the fragmentation is at around 4%.

However, each records will be updated at least 3 times afterwards. This generates a fragmentation of this clustered index of more than 99% (with default Fill factor)..

Or with other fill factors I tested:
– Setting a Fill factor of 80 on the clustered index (Fragmentation > 98%)
– Setting a Fill factor of 50 on the clustered index (Fragmentation > 94%)

So this doesn't look like it would help a lot…

I tested each fill factor setting on a newly created table with 2000 inserts (and the 3+ updates)

there is no select or update which does not include the primary key in the query.

Has anybody an idea why this index has such a high fragmentation?

Updates are made through ADO.NET this is the generated command (traced by the SQL Profiler):

exec sp_executesql N'UPDATE MyTable SET MyValue = @MyValue WHERE MyId = @MyId',
N'@MyValue varchar(50),@MyId int',@MyValue='This is some random value',@MyId=1234

Other updates follow the same pattern (update other values, but alway providing the @MyId parameter)

Thank you for your help 🙂

Best Answer

FILLFACTOR only applies when you build or rebuild an index, not during normal operation. Normal operations always try to fill the pages to 100%.

If you insert a row that has a variable width, then update the row to be longer, that row will no longer fit on the page if there isn't enough extra space to store the after-image on the same page. If there isn't enough space, this will cause a page split, which is the process that creates the necessary space.

What's a bit misleading about page splits is that there are "good splits" and "bad splits," even though the performance counter counts all of them.

  • A good split is when new rows are added to the end of the index, like what happens when you do your initial batch of INSERTs. The new row doesn't fit on the last page, so the storage engine must allocate a new page and logically hook it up to the last index page. The new page will probably exist physically after the old-last index page.

  • A bad split is when a page must be inserted in the middle of an index: the new page is attached to the index structure logically on both sides, but may (likely) not exist in contiguous physical order to those pages.

Fragmentation is the discrepancy between the logical and physical order, and for the most part, only the bad type of page splits cause fragmentation.

Because you're inflating the size of already-existing rows, this causes the bad type of page split, which is why you're seeing high fragmentation numbers.

It's unclear what your exact process is, and how many rows are ultimately going to end up in this table. If it's something like a one-time population, do the full population process, then turn around and rebuild the clustered index with 100% FILLFACTOR.

If this process happens continuously, you could "pre-allocate" the space by doing the following: add a dummy variable-width column to the table, populate it to the max length on INSERT, and then on the first UPDATE, set the dummy value to NULL while updating the real value. This method will probably add overhead to the logging mechanism because of all the data values flying around.

In general, though, you only need to concern yourself with fragmentation when all of these factors are true:

  • The index is large enough that randomly reading the pages (extents) is too slow (choose your own definition of what "too slow" means)
  • The index will be scanned
  • The index pages (extents) will be physically read from disk (i.e., they are not already in the buffer pool)
  • The storage subsystem is poor at handling random reads (i.e., spindle-based, and the bytes won't come out of cache)

I would recommend, however, putting in place a solid index maintenance solution to keep things more or less in check. You can search this site for recommendations with regards to that.