SELECT COUNT(*)查询是否必须执行全表扫描?

Wat*_* v2 2 database sql-server sql-server-2014

获取表中所有行计数的查询是否必须执行全表扫描,或者SQL Server是否在某处保留了行数?

SELECT COUNT(*) FROM TABLE_NAME;
Run Code Online (Sandbox Code Playgroud)

该表TABLE_NAME有一个主键,因此是一个聚簇索引,如下所示:

CREATE TABLE TABLE_NAME
(
  Id int PRIMARY KEY IDENTITY(1, 1),
  Name nvarchar(50) NOT NULL
);
Run Code Online (Sandbox Code Playgroud)

我正在使用Microsoft SQL Server 2014.

Man*_*ano 5

当 SQL Server 执行类似 的查询时SELECT COUNT(*),SQL Server 将使用最窄的non-clustered index来计算行数。如果该表没有任何non-clustered index,则必须扫描该表。

如果你的桌子上有一个,clustered index你可以更快地得到你的计数。


EzL*_*zLo 5

服务器将始终读取所有记录(如果有索引,则它将扫描整个索引)以计算行数.只要你这样做,你就无法摆脱这种情况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
Run Code Online (Sandbox Code Playgroud)

如果您要查找记录的大致计数,您还可以使用以下查询:

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
Run Code Online (Sandbox Code Playgroud)

这将显示非系统表的计算(非精确)行数和数据大小(它们可能具有的任何索引)的总和,相对快速而不检索记录.