WITH
TB AS(
SELECT
TB.object_id,
schema_name = SCH.name,
table_name = TB.name
FROM sys.tables TB
INNER JOIN sys.schemas SCH
ON TB.schema_id = SCH.schema_id
WHERE is_ms_shipped = 0 -- 此条件表示仅查询不是由内部 SQL Server 组件创建对象
),
PS AS(
-- 此部分计算表空间的信息
SELECT
object_id,
reserved_pages = SUM(reserved_page_count),
used_pages = SUM(used_page_count),
pages = SUM(
CASE
WHEN index_id > 1 THEN lob_used_page_count + row_overflow_used_page_count
ELSE in_row_data_page_count + lob_used_page_count + row_overflow_used_page_count
END),
row_count = SUM (
CASE
WHEN index_id < 2 THEN row_count
ELSE 0
END)
FROM sys.dm_db_partition_stats PS
GROUP BY object_id
),
ITPS AS(
-- 此部分计算包含 XML INDEX 和 FULLTEXT INDEXE 的空间信息(如果有的话)
SELECT
object_id = ITB.parent_id,
reserved_pages = SUM(reserved_page_count),
used_pages = SUM(used_page_count)
FROM sys.dm_db_partition_stats P
INNER JOIN sys.internal_tables ITB
ON P.object_id = ITB.object_id
WHERE ITB.internal_type IN(202, 204)
GROUP BY ITB.parent_id
),
SIZE AS(
-- 此部分合并所有的空间信息
SELECT
PS.object_id,
reserved_pages = PS.reserved_pages + ISNULL(ITPS.reserved_pages, 0),
used_pages = PS.used_pages + ISNULL(ITPS.used_pages, 0),
PS.pages,
PS.row_count
FROM PS
LEFT JOIN ITPS
ON PS.object_id = ITPS.object_id
)
-- 显示最终的空间统计结果
-- 在前面的统计中,空间统计以页为单位, 8K/页,最终的统计将页数*8,得到KB为单位的空间大小
SELECT
TB.schema_name,
TB.table_name,
SIZE.row_count,
reserved = SIZE.reserved_pages * 8,
data = SIZE.pages * 8,
index_size = CASE
WHEN SIZE.used_pages > SIZE.pages
THEN SIZE.used_pages - SIZE.pages
ELSE 0
END * 8,
unused = CASE
WHEN SIZE.reserved_pages > SIZE.used_pages
THEN SIZE.reserved_pages - SIZE.used_pages
ELSE 0
END * 8
FROM TB
INNER JOIN SIZE
ON TB.object_id = SIZE.object_id
ORDER BY schema_name,data desc