Должен ли запрос SELECT COUNT (*) выполнять полное сканирование таблицы? - PullRequest
0 голосов
/ 28 февраля 2019

Должен ли запрос, который получает количество всех строк в таблице, выполнить полное сканирование таблицы или SQL Server где-то поддерживает количество строк?

SELECT COUNT(*) FROM TABLE_NAME;

Таблица TABLE_NAME имеетпервичный ключ и, следовательно, кластеризованный индекс, и выглядит так:

CREATE TABLE TABLE_NAME
(
  Id int PRIMARY KEY IDENTITY(1, 1),
  Name nvarchar(50) NOT NULL
);

Я использую Microsoft SQL Server 2014.

Ответы [ 4 ]

0 голосов
/ 28 февраля 2019

Сервер всегда будет читать все записи (если есть индекс, то он будет сканировать весь индекс) для подсчета строк.Вы не можете избежать этого, пока выполняете SELECT COUNT(*) FROM Table.

Если ваша таблица имеет кластерный индекс, вы можете изменить свой запрос на запрос «под капотом» наполучить счетчик без фактического извлечения записей с помощью:

SELECT OBJECT_NAME(i.id) [Table_Name], i.rowcnt [Row_Count]
FROM sys.sysindexes i WITH (NOLOCK)
WHERE i.indid in (0,1)
ORDER BY i.rowcnt desc

, если вы ищете приблизительное количество записей, вы также можете использовать следующий запрос:

SELECT 
    TableName = t.NAME,
    SchemaName = s.Name,
    [RowCount] = p.rows,
    TotalSpaceMB = CONVERT(DECIMAL(18,2), SUM(a.total_pages) * 8 / 1024.0), 
    UsedSpaceMB = CONVERT(DECIMAL(18,2), SUM(a.used_pages) * 8 / 1024.0),
    UnusedSpaceMB = CONVERT(DECIMAL(18,2), (SUM(a.total_pages) - SUM(a.used_pages)) * 8 / 1024.0)
FROM 
    sys.tables t
    INNER JOIN sys.indexes i ON t.OBJECT_ID = i.object_id
    INNER JOIN sys.partitions p ON i.object_id = p.OBJECT_ID AND i.index_id = p.index_id
    INNER JOIN sys.allocation_units a ON p.partition_id = a.container_id
    LEFT OUTER JOIN sys.schemas s ON t.schema_id = s.schema_id
WHERE 
    t.NAME NOT LIKE 'dt%' 
    AND t.is_ms_shipped = 0
    AND i.OBJECT_ID > 255 
GROUP BY 
    t.Name, 
    s.Name, 
    p.Rows
ORDER BY 
    TotalSpaceMB DESC

Это покажет несистемные таблицы с их рассчитанным (не точным) количеством строк и суммой размеров их данных (с любым индексом, который у них может быть), относительно быстро без извлечения записей.

0 голосов
/ 28 февраля 2019
SELECT COUNT(*) FROM TABLE_NAME;

Полное сканирование таблицы.

Для оптимизации вы можете обратиться к this .

0 голосов
/ 28 февраля 2019

вы можете следующим образом.это лучше по производительности, я думаю.

SELECT COUNT(1) FROM TABLE_NAME 
0 голосов
/ 28 февраля 2019

Когда SQL Server выполняет запрос, подобный SELECT COUNT(*), SQL Server будет использовать самый узкий non-clustered index для подсчета строк.Если в таблице нет non-clustered index, она должна будет отсканировать таблицу.

Если в вашей таблице clustered index, вы можете получить свой счет еще быстрее.

...