获取数据库中所有表的大小

Get size of all tables in database

提问人:Eric 提问时间:10/26/2011 最后编辑:Mathieu RendaEric 更新时间:10/12/2023 访问量:2091732

问:

我继承了一个相当大的SQL Server数据库。考虑到它包含的数据,它似乎占用的空间比我预期的要多。

有没有一种简单的方法来确定每个表占用了多少磁盘空间?

sql-server t-sql

评论

1赞 Rob Allen 10/26/2011
您可以访问哪些角色?您是 DBA,还是通过 Web 主机、客户端或类似设备进行管理?
2赞 Joe Stefanelli 10/26/2011
SQL Server 中表和索引大小的可能重复
1赞 Eric 10/26/2011
@RobAllen我对数据库具有完全访问权限,因此需要任何角色的脚本就足够了。
2赞 naXa stands with Ukraine 3/17/2017
如何判断哪些表在 SQL Server 2005 数据库中占用的空间最多?
0赞 Mark Meuer 2/22/2023
在我看来,可以在这个问题的这个(我的)答案(stackoverflow.com/a/3719990/9117)中找到更干净的解决方案(stackoverflow.com/questions/3606366/......

答:

3419赞 marc_s 10/26/2011 #1
SELECT 
    t.name AS TableName,
    s.name AS SchemaName,
    p.rows,
    SUM(a.total_pages) * 8 AS TotalSpaceKB, 
    CAST(ROUND(((SUM(a.total_pages) * 8) / 1024.00), 2) AS NUMERIC(36, 2)) AS TotalSpaceMB,
    SUM(a.used_pages) * 8 AS UsedSpaceKB, 
    CAST(ROUND(((SUM(a.used_pages) * 8) / 1024.00), 2) AS NUMERIC(36, 2)) AS UsedSpaceMB, 
    (SUM(a.total_pages) - SUM(a.used_pages)) * 8 AS UnusedSpaceKB,
    CAST(ROUND(((SUM(a.total_pages) - SUM(a.used_pages)) * 8) / 1024.00, 2) AS NUMERIC(36, 2)) AS UnusedSpaceMB
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, t.name

评论

12赞 GEMI 12/7/2011
愚蠢的问题,但是此查询是否可能导致行锁定?
11赞 Jens Frandsen 7/12/2013
索引也使用空间,索引使用的空间量可以通过下面的查询找到。
10赞 Akos Lukacs 11/12/2013
您的脚本在筛选索引方面存在问题:对于给定表的每个筛选索引,我在结果中看到带有该表名称的额外行。每个额外行的“RowCounts”对应于其中一个筛选索引所覆盖的行数。(在 Sql2012 上)
18赞 PRMan 4/23/2014
如果表已分区,则它们会多次显示,而没有任何指示。您可以将p.partition_number添加到选择列表中,也可以 SUM(p.Rows) 并将其从组中删除。
10赞 Udit Solanki 9/5/2018
对于由于过滤索引而面临重复行的任何人(谁需要实际的表行),只需再添加一个 where 条件。AND i.filter_definition IS NULL
49赞 Eric 10/26/2011 #2

经过一番搜索,我找不到一种简单的方法来获取所有表格的信息。有一个名为 sp_spaceused 的方便存储过程,它将返回数据库使用的所有空间。如果提供表名,它将返回该表使用的空间。但是,存储过程返回的结果不可排序,因为列是字符值。

以下脚本将生成我正在寻找的信息。

create table #TableSize (
    Name varchar(255),
    [rows] int,
    reserved varchar(255),
    data varchar(255),
    index_size varchar(255),
    unused varchar(255))
create table #ConvertedSizes (
    Name varchar(255),
    [rows] int,
    reservedKb int,
    dataKb int,
    reservedIndexSize int,
    reservedUnused int)

EXEC sp_MSforeachtable @command1="insert into #TableSize
EXEC sp_spaceused '?'"
insert into #ConvertedSizes (Name, [rows], reservedKb, dataKb, reservedIndexSize, reservedUnused)
select name, [rows], 
SUBSTRING(reserved, 0, LEN(reserved)-2), 
SUBSTRING(data, 0, LEN(data)-2), 
SUBSTRING(index_size, 0, LEN(index_size)-2), 
SUBSTRING(unused, 0, LEN(unused)-2)
from #TableSize

select * from #ConvertedSizes
order by reservedKb desc

drop table #TableSize
drop table #ConvertedSizes
59赞 Royi Namir 10/26/2011 #3
 exec  sp_spaceused N'dbo.MyTable'

对于所有表,请使用。.(根据保罗的评论添加)

exec sp_MSForEachTable 'exec sp_spaceused [?]'

评论

6赞 marc_s 10/26/2011
鬼鬼祟祟 - 你从哪个不显示任何关于表格的内容,改为哪个 - 但一次只显示一个表格......它不会让您大致了解您拥有哪些表、它们有多少行以及它们占用了多少空间。exec sp_helpdbexec sp_spaceused
5赞 Paul 4/15/2013
exec sp_MSForEachTable 'exec sp_spaceused [?]'
0赞 wilmol 5/30/2023
有没有办法对每个表的结果求和?
5赞 William Walseth 10/26/2011 #4

这将为您提供每个表的大小和记录计数。

set ANSI_NULLS ON
set QUOTED_IDENTIFIER ON
GO
-- Get a list of tables and their sizes on disk
ALTER PROCEDURE [dbo].[sp_Table_Sizes]
AS
BEGIN
    -- SET NOCOUNT ON added to prevent extra result sets from
    -- interfering with SELECT statements.
    SET NOCOUNT ON;
DECLARE @table_name VARCHAR(500)  
DECLARE @schema_name VARCHAR(500)  
DECLARE @tab1 TABLE( 
        tablename VARCHAR (500) collate database_default 
       ,schemaname VARCHAR(500) collate database_default 
) 

CREATE TABLE #temp_Table ( 
        tablename sysname 
       ,row_count INT 
       ,reserved VARCHAR(50) collate database_default 
       ,data VARCHAR(50) collate database_default 
       ,index_size VARCHAR(50) collate database_default 
       ,unused VARCHAR(50) collate database_default  
) 

INSERT INTO @tab1  
SELECT Table_Name, Table_Schema  
FROM information_schema.tables  
WHERE TABLE_TYPE = 'BASE TABLE' 

DECLARE c1 CURSOR FOR 
SELECT Table_Schema + '.' + Table_Name   
FROM information_schema.tables t1  
WHERE TABLE_TYPE = 'BASE TABLE' 

OPEN c1 
FETCH NEXT FROM c1 INTO @table_name 
WHILE @@FETCH_STATUS = 0  
BEGIN   
        SET @table_name = REPLACE(@table_name, '[','');  
        SET @table_name = REPLACE(@table_name, ']','');  

        -- make sure the object exists before calling sp_spacedused 
        IF EXISTS(SELECT id FROM sysobjects WHERE id = OBJECT_ID(@table_name)) 
        BEGIN 
               INSERT INTO #temp_Table EXEC sp_spaceused @table_name, false; 
        END 

        FETCH NEXT FROM c1 INTO @table_name 
END 
CLOSE c1 
DEALLOCATE c1 

SELECT  t1.* 
       ,t2.schemaname  
FROM #temp_Table t1  
INNER JOIN @tab1 t2 ON (t1.tablename = t2.tablename ) 
ORDER BY schemaname,t1.tablename; 

DROP TABLE #temp_Table
END

评论

2赞 marc_s 10/26/2011
如果您发布代码、XML 或数据示例,请在文本编辑器中突出显示这些行,然后单击编辑器工具栏上的“代码示例”按钮 ( ) 以很好地格式化和语法突出显示它!{ }
133赞 Axle 10/26/2011 #5

sp_spaceused可以获取有关表、索引视图或整个数据库使用的磁盘空间的信息。

例如:

USE MyDatabase; GO

EXEC sp_spaceused N'User.ContactInfo'; GO

这将报告 ContactInfo 表的磁盘使用情况信息。

要一次将其用于所有表,请执行以下操作:

USE MyDatabase; GO

sp_msforeachtable 'EXEC sp_spaceused [?]' GO

还可以从 SQL Server 的右键单击“标准报表”功能中获取磁盘使用情况。若要访问此报表,请从对象资源管理器中的服务器对象导航,向下移动到“数据库”对象,然后右键单击任何数据库。从显示的菜单中,选择“报告”,然后选择“标准报告”,然后选择“按分区划分的磁盘使用情况:[DatabaseName]”。

评论

4赞 syneticon-dj 12/23/2014
这很简洁,尽管在 SSMS 中使用 很容易触发 ,如果你有大量表,因此使用临时表来存储结果可能是一个更好的主意。sp_msforeachtableSystem.OutOfMemoryException
3赞 DarthPablo 8/19/2015
我可以看到sp_spacedused的主要问题是它似乎以人类可读的格式返回数据(例如,在我的情况下,在“保留”列中,它有“152 KB”)。我认为这将酌情切换到 MB/GB。这在很多情况下显然很有用,但如果您需要根据大小应用一些逻辑,或者想要比较值或其他什么,则不然。我寻找一种关闭它的方法,但我找不到(我使用的是SQL Server 2005:()
0赞 Goku - stands with Palestine 8/22/2023
sp_msforeachtable在 SQL Azure 中不可用...
0赞 FranTastic 10/11/2023
简明扼要,但有效。
1赞 Andrew 5/21/2012 #6

我的文章仅与 SQL Server 2000 相关,并且已经过测试,可以在我的环境中工作。

此代码访问单个实例的所有可能的数据库,而不仅仅是单个数据库。

我使用两个临时表来帮助收集适当的数据,然后将结果转储到一个“实时”表中。

返回的数据是:DatabaseName、DatabaseTableName、Rows(在表中)、data(表的大小(以 KB 为单位)、条目数据(我发现这对于了解我上次运行脚本的时间很有用)。

此代码的缺点是“data”字段未存储为 int(字符“KB”保留在该字段中),这对于排序很有用(但不是完全必要)。

希望这段代码能帮助到外面的人,为他们节省一些时间!

CREATE PROCEDURE [dbo].[usp_getAllDBTableSizes]

AS
BEGIN
   SET NOCOUNT OFF

   CREATE TABLE #DatabaseTables([dbname] sysname,TableName sysname)
   CREATE TABLE #AllDatabaseTableSizes(Name sysname,[rows] VARCHAR(18), reserved VARCHAR(18), data VARCHAR(18), index_size VARCHAR(18), unused VARCHAR(18))

   DECLARE @SQL nvarchar(4000)
   SET @SQL='select ''?'' AS [Database], Table_Name from [?].information_schema.tables WHERE TABLE_TYPE = ''BASE TABLE'' '

   INSERT INTO #DatabaseTables(DbName, TableName)
      EXECUTE sp_msforeachdb @Command1=@SQL

   DECLARE AllDatabaseTables CURSOR LOCAL READ_ONLY FOR   
   SELECT TableName FROM #DatabaseTables

   DECLARE AllDatabaseNames CURSOR LOCAL READ_ONLY FOR   
   SELECT DBName FROM #DatabaseTables

   DECLARE @DBName sysname  
   OPEN AllDatabaseNames  

   DECLARE @TName sysname
   OPEN AllDatabaseTables  

   WHILE 1=1 BEGIN 
      FETCH NEXT FROM AllDatabaseNames INTO @DBName  
      FETCH NEXT FROM AllDatabaseTables INTO @TName 
      IF @@FETCH_STATUS<>0 BREAK  
      INSERT INTO #AllDatabaseTableSizes
         EXEC ( 'EXEC ' + @DBName + '.dbo.sp_spaceused ' + @TName) 

   END 

   --http://msdn.microsoft.com/en-us/library/aa175920(v=sql.80).aspx
   INSERT INTO rsp_DatabaseTableSizes (DatabaseName, name, [rows], data)
      SELECT   [dbname], name, [rows],  data FROM #DatabaseTables
      INNER JOIN #AllDatabaseTableSizes
      ON #DatabaseTables.TableName = #AllDatabaseTableSizes.Name
      GROUP BY [dbname] , name, [rows],  data
      ORDER BY [dbname]
   --To be honest, I have no idea what exact duplicates we are dropping
    -- but in my case a near enough approach has been good enough.
   DELETE FROM [rsp_DatabaseTableSizes]
   WHERE name IN 
      ( 
      SELECT name 
      FROM [rsp_DatabaseTableSizes]
      GROUP BY name
      HAVING COUNT(*) > 1
      )

   DROP TABLE #DatabaseTables
   DROP TABLE #AllDatabaseTableSizes

   CLOSE AllDatabaseTables  
   DEALLOCATE AllDatabaseTables  

   CLOSE AllDatabaseNames  
   DEALLOCATE AllDatabaseNames      
END

--EXEC [dbo].[usp_getAllDBTableSizes] 

如果您需要知道,rsp_DatabaseTableSizes表是通过以下方式创建的:

CREATE TABLE [dbo].[rsp_DatabaseSizes](
    [DatabaseName] [varchar](1000) NULL,
    [dbSize] [decimal](15, 2) NULL,
    [DateUpdated] [smalldatetime] NULL
) ON [PRIMARY]

GO
769赞 Kevin Brydon 4/24/2013 #7

如果您使用的是 SQL Server Management Studio (SSMS),则可以运行标准报表,而不是运行查询(在本例中返回重复行)。

  1. 右键单击数据库
  2. 导航到“报告”>“标准报告”>按表显示磁盘使用情况

注意:数据库兼容级别必须设置为 90 或更高才能正常工作。查看 http://msdn.microsoft.com/en-gb/library/bb510680.aspx

注意:使用 Azure SQL 数据库时,此报表不可用

评论

68赞 ValGe 1/13/2014
在 Management Studio 2012 中,可以执行以下操作:视图-对象资源管理器详细信息 (F7) 并导航到对象资源管理器中的“表”。在“详细信息”中,右键单击标题,然后选择“大小列”。
1赞 Simon_Weaver 2/23/2017
您甚至可以通过单击标题进行排序,甚至可以导出到 Excel(我这样做是为了获得平均行大小)
0赞 SixOThree 4/16/2020
这些值是以 KB 还是 KiB 为单位?在我正在处理的规模上,差异是有意义的。我知道许多产品在实际上是 KiB 时列出了 KB。
0赞 TylerH 2/24/2022
@SixOThree 由于 SSMS 说的是“KB”而不是“KiB”,因此我们必须假设它实际上意味着“KB”。我想,您可以自己将所有表的值相加,并将其与文件资源管理器中 .mdf 文件的大小进行比较。
0赞 JeeShen Lee 2/14/2023
我找不到“按表划分的磁盘使用情况”选项。我正在使用 SSMS 19 和 SQL Azure 数据库。
39赞 Jens Frandsen 7/12/2013 #8

上面的查询适用于查找表(包括索引)使用的空间量,但如果要比较表上的索引使用了多少空间,请使用以下查询:

SELECT
    OBJECT_NAME(i.OBJECT_ID) AS TableName,
    i.name AS IndexName,
    i.index_id AS IndexID,
    8 * SUM(a.used_pages) AS 'Indexsize(KB)'
FROM
    sys.indexes AS i
    JOIN sys.partitions AS p ON p.OBJECT_ID = i.OBJECT_ID AND p.index_id = i.index_id
    JOIN sys.allocation_units AS a ON a.container_id = p.partition_id
WHERE
    i.is_primary_key = 0 -- fix for size discrepancy
GROUP BY
    i.OBJECT_ID,
    i.index_id,
    i.name
ORDER BY
    OBJECT_NAME(i.OBJECT_ID),
    i.index_id

评论

1赞 Derek 1/10/2019
对特定表的 Indexsize(KB) 列求和与sp_spaceused index_size不一致的原因是什么?
1赞 CodeAngry 1/11/2019
@Derek 通过添加 .现在尺寸应该匹配。where [i].[is_primary_key] = 0
0赞 Derek 1/11/2019
谢谢,但这实际上也没有成功。我有一个(非常小的)测试数据库,感兴趣的表有两个索引 - 一列上的主聚簇索引和另外两列上的非聚簇索引。此查询显示它们中的每一个都使用 16kB,但sp_spaceused表示总索引使用量为 24kB。我的部分困惑是:将此查询与已接受答案的“UsedSpaceKB”进行比较,我看不出真正的区别。相同的连接,只是缺少 sys.tables 的添加。是我遗漏了什么,还是这个查询本身就坏了?
1赞 CodeAngry 1/14/2019
我有大型数据库。并且尺寸与 .我测量 GB,所以一些不匹配的 megs 并不多。我不在乎确切的尺寸,只是一个想法。sp_spaceused
3赞 Alan Cardoso 9/7/2013 #9

我marc_s答案之上又添加了几列:

with fs
as
(
select i.object_id,
        p.rows AS RowCounts,
        SUM(a.total_pages) * 8 AS TotalSpaceKb
from     sys.indexes i 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
WHERE 
    i.OBJECT_ID > 255 
GROUP BY 
    i.object_id,
    p.rows
)

SELECT 
    t.NAME AS TableName,
    fs.RowCounts,
    fs.TotalSpaceKb,
    t.create_date,
    t.modify_date,
    ( select COUNT(1)
        from sys.columns c 
        where c.object_id = t.object_id ) TotalColumns    
FROM 
    sys.tables t INNER JOIN      
    fs  ON t.OBJECT_ID = fs.object_id
WHERE 
    t.NAME NOT LIKE 'dt%' 
    AND t.is_ms_shipped = 0
ORDER BY 
    t.Name
10赞 Mark 11/12/2013 #10
-- Show the size of all the tables in a database sort by data size descending
SET NOCOUNT ON
DECLARE @TableInfo TABLE (tablename varchar(255), rowcounts int, reserved varchar(255), DATA varchar(255), index_size varchar(255), unused varchar(255))
DECLARE @cmd1 varchar(500)
SET @cmd1 = 'exec sp_spaceused ''?'''

INSERT INTO @TableInfo (tablename,rowcounts,reserved,DATA,index_size,unused)
EXEC sp_msforeachtable @command1=@cmd1

SELECT * FROM @TableInfo ORDER BY Convert(int,Replace(DATA,' KB','')) DESC
17赞 sqladmin 1/31/2014 #11

如果需要计算 SSMS 中“表属性 - 存储”页上完全相同的数字,则需要使用与 SSMS 中相同的方法对它们进行计数(适用于 sql server 2005 及更高版本...并且也适用于具有 LOB 字段的表 - 因为仅计算“used_pages”不足以显示准确的索引大小):

;with cte as (
SELECT
t.name as TableName,
SUM (s.used_page_count) as used_pages_count,
SUM (CASE
            WHEN (i.index_id < 2) THEN (in_row_data_page_count + lob_used_page_count + row_overflow_used_page_count)
            ELSE lob_used_page_count + row_overflow_used_page_count
        END) as pages
FROM sys.dm_db_partition_stats  AS s 
JOIN sys.tables AS t ON s.object_id = t.object_id
JOIN sys.indexes AS i ON i.[object_id] = t.[object_id] AND s.index_id = i.index_id
GROUP BY t.name
)
select
    cte.TableName, 
    cast((cte.pages * 8.)/1024 as decimal(10,3)) as TableSizeInMB, 
    cast(((CASE WHEN cte.used_pages_count > cte.pages 
                THEN cte.used_pages_count - cte.pages
                ELSE 0 
          END) * 8./1024) as decimal(10,3)) as IndexSizeInMB
from cte
order by 2 desc

评论

0赞 arthur 1/29/2021
您能详细说明为什么给出堆的页数吗?中只有一个索引满足此条件(该索引是主键:i.index_id = 1)。但是,如果我将大小与sb_spaceused结果进行比较,那么它们(在很大程度上)匹配WHEN (i.index_id < 2|) THEN (in_row_data_page_count + lob_used_page_count + row_overflow_used_page_count)sql server 2014
4赞 user4658783 3/11/2015 #12

从使用 OSQL 的命令提示符:

OSQL -E -d <*databasename*> -Q "exec sp_msforeachtable 'sp_spaceused [?]'" > result.txt
6赞 Ardalan Shahgholi 4/25/2015 #13

要获取一个数据库中的所有表大小,您可以使用以下查询:

Exec sys.sp_MSforeachtable ' sp_spaceused "?" '

您可以更改它以将所有结果插入临时表,然后从临时表中进行选择。

Insert into #TempTable Exec sys.sp_MSforeachtable ' sp_spaceused "?" ' 
Select * from #TempTable
1赞 Chris Smith 10/8/2015 #14

根据上面@Mark答案,添加了 @updateusage='true' 以强制执行最新的大小统计数据 (https://msdn.microsoft.com/en-us/library/ms188776.aspx):

        SET NOCOUNT ON
        DECLARE @TableInfo TABLE (tablename varchar(255), rowcounts int, reserved varchar(255), DATA varchar(255), index_size varchar(255), unused varchar(255))
        DECLARE @cmd1 varchar(500)
        SET @cmd1 = 'exec sp_spaceused @objname =''?'', @updateusage =''true'' '

        INSERT INTO @TableInfo (tablename,rowcounts,reserved,DATA,index_size,unused)
        EXEC sp_msforeachtable @command1=@cmd1 
SELECT * FROM @TableInfo ORDER BY Convert(int,Replace(DATA,' KB','')) DESC
17赞 Joel Harkes 4/13/2016 #15

与Marc_s的答案相比,这是一个小小的变化,因为我经常回到这个页面,按大多数行的第一行排序:

SELECT
    t.NAME AS TableName,
    s.Name AS SchemaName,
    p.rows AS RowCounts,
    SUM(a.total_pages) * 8 AS TotalSpaceKB,
    SUM(a.used_pages) * 8 AS UsedSpaceKB,
    (SUM(a.total_pages) - SUM(a.used_pages)) * 8 AS UnusedSpaceKB
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
    --p.rows DESC --Uncomment to order by amount rows instead of size in KB.
    SUM(a.total_pages) DESC 
16赞 xav 4/14/2016 #16

我们使用表分区,由于重复记录,上面提供的查询遇到了一些问题。

对于需要此功能的用户,可以在下面找到 SQL Server 2014 在生成“按表划分的磁盘使用情况”报告时运行的查询。我认为它也适用于以前版本的 SQL Server。

它就像一个魅力。

SELECT
    a2.name AS [tablename],
    a1.rows as row_count,
    (a1.reserved + ISNULL(a4.reserved,0))* 8 AS reserved, 
    a1.data * 8 AS data,
    (CASE WHEN (a1.used + ISNULL(a4.used,0)) > a1.data THEN (a1.used + ISNULL(a4.used,0)) - a1.data ELSE 0 END) * 8 AS index_size,
    (CASE WHEN (a1.reserved + ISNULL(a4.reserved,0)) > a1.used THEN (a1.reserved + ISNULL(a4.reserved,0)) - a1.used ELSE 0 END) * 8 AS unused
FROM
    (SELECT 
        ps.object_id,
        SUM (
            CASE
                WHEN (ps.index_id < 2) THEN row_count
                ELSE 0
            END
            ) AS [rows],
        SUM (ps.reserved_page_count) AS reserved,
        SUM (
            CASE
                WHEN (ps.index_id < 2) THEN (ps.in_row_data_page_count + ps.lob_used_page_count + ps.row_overflow_used_page_count)
                ELSE (ps.lob_used_page_count + ps.row_overflow_used_page_count)
            END
            ) AS data,
        SUM (ps.used_page_count) AS used
    FROM sys.dm_db_partition_stats ps
        WHERE ps.object_id NOT IN (SELECT object_id FROM sys.tables WHERE is_memory_optimized = 1)
    GROUP BY ps.object_id) AS a1
LEFT OUTER JOIN 
    (SELECT 
        it.parent_id,
        SUM(ps.reserved_page_count) AS reserved,
        SUM(ps.used_page_count) AS used
     FROM sys.dm_db_partition_stats ps
     INNER JOIN sys.internal_tables it ON (it.object_id = ps.object_id)
     WHERE it.internal_type IN (202,204)
     GROUP BY it.parent_id) AS a4 ON (a4.parent_id = a1.object_id)
INNER JOIN sys.all_objects a2  ON ( a1.object_id = a2.object_id ) 
INNER JOIN sys.schemas a3 ON (a2.schema_id = a3.schema_id)
WHERE a2.type <> N'S' and a2.type <> N'IT'
ORDER BY a3.name, a2.name
9赞 Anjan Kant 5/2/2016 #17

下面是通过以下步骤快速获取所有表大小的方法:

  1. 编写给定的 T-SQL 命令以列出所有数据库表:

    select 'exec sp_spaceused ' + TABLE_NAME from INFORMATION_SCHEMA.TABLES where TABLE_TYPE = 'BASE TABLE'
    
  2. 现在复制数据库表列表,并将其复制到新的查询分析器窗口中

    exec sp_spaceused table1
    exec sp_spaceused table2
    exec sp_spaceused table3
    exec sp_spaceused table4
    exec sp_spaceused table5
    
  3. 在 SQL 查询分析器中,从顶部工具栏选项“结果到文件”( + + )。CtrlShiftF

  4. 现在终于打 执行 按钮从上面的工具栏中标记为红色。

  5. 所有表的数据库大小现在都存储在计算机上的文件中。

    Enter image description here

1赞 Zach Smith 10/11/2016 #18

作为对 marc_s 的答案(已被接受的答案)的简单扩展,它被调整为返回列数并允许过滤:

SELECT *
FROM
(

SELECT 
    t.NAME AS TableName,
    s.Name AS SchemaName,
    p.rows AS RowCounts,
    COUNT(DISTINCT c.COLUMN_NAME) as ColumnCount,
    SUM(a.total_pages) * 8 AS TotalSpaceKB, 
    (SUM(a.used_pages) * 8) AS UsedSpaceKB, 
    (SUM(a.total_pages) - SUM(a.used_pages)) * 8 AS UnusedSpaceKB
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
INNER JOIN
    INFORMATION_SCHEMA.COLUMNS c ON t.NAME = c.TABLE_NAME
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
) AS Result

WHERE
    RowCounts > 1000
    AND ColumnCount > 10
ORDER BY 
    UsedSpaceKB DESC

评论

0赞 dreamca4er 10/31/2019
联接 Columns 表后,您将不再拥有正确的表空间。外部应用将是修复。
76赞 Sparrow 1/19/2018 #19

下面是另一种方法:使用 SQL Server Management Studio,在对象资源管理器中,转到数据库并选择“表”

enter image description here

然后打开“对象资源管理器详细信息”(通过按 F7 或转到“视图”->“对象资源管理器详细信息”)。在对象资源管理器详细信息页中,右键单击列标题并启用要在页面中看到的列。您也可以按任何列对数据进行排序。

enter image description here

评论

0赞 pcdev 5/16/2019
@batmaci 当你对 Azure SQL 数据库发表评论时,不确定这是否有效,但它现在似乎至少在最新版本的 SSMS 中部分有效。对我来说,对表元数据的查询似乎超时了,但在它超时之前,它似乎返回了一些 (3-10) 个表,包括(可靠地)选定的表。选择一个表,然后单击“刷新”以查看所需的表(如果未显示)。
0赞 Max 5/28/2020
您还可以使用 NirSoft SysExporter 等实用程序将列表导出为 CSV 文件: nirsoft.net/utils/sysexp.html
0赞 Irf 8/11/2022
对于 Azure,这应该有效
24赞 Santhoshkumar KB 4/24/2018 #20

扩展@xav处理表分区以获取大小(以 MB 和 GB 为单位)的答案。 在 SQL Server 2008/2012 上测试(注释了一行,其中is_memory_optimized = 1)

SELECT
    a2.name AS TableName,
    a1.rows as [RowCount],
    --(a1.reserved + ISNULL(a4.reserved,0)) * 8 AS ReservedSize_KB,
    --a1.data * 8 AS DataSize_KB,
    --(CASE WHEN (a1.used + ISNULL(a4.used,0)) > a1.data THEN (a1.used + ISNULL(a4.used,0)) - a1.data ELSE 0 END) * 8 AS IndexSize_KB,
    --(CASE WHEN (a1.reserved + ISNULL(a4.reserved,0)) > a1.used THEN (a1.reserved + ISNULL(a4.reserved,0)) - a1.used ELSE 0 END) * 8 AS UnusedSize_KB,
    CAST(ROUND(((a1.reserved + ISNULL(a4.reserved,0)) * 8) / 1024.00, 2) AS NUMERIC(36, 2)) AS ReservedSize_MB,
    CAST(ROUND(a1.data * 8 / 1024.00, 2) AS NUMERIC(36, 2)) AS DataSize_MB,
    CAST(ROUND((CASE WHEN (a1.used + ISNULL(a4.used,0)) > a1.data THEN (a1.used + ISNULL(a4.used,0)) - a1.data ELSE 0 END) * 8 / 1024.00, 2) AS NUMERIC(36, 2)) AS IndexSize_MB,
    CAST(ROUND((CASE WHEN (a1.reserved + ISNULL(a4.reserved,0)) > a1.used THEN (a1.reserved + ISNULL(a4.reserved,0)) - a1.used ELSE 0 END) * 8 / 1024.00, 2) AS NUMERIC(36, 2)) AS UnusedSize_MB,
    --'| |' Separator_MB_GB,
    CAST(ROUND(((a1.reserved + ISNULL(a4.reserved,0)) * 8) / 1024.00 / 1024.00, 2) AS NUMERIC(36, 2)) AS ReservedSize_GB,
    CAST(ROUND(a1.data * 8 / 1024.00 / 1024.00, 2) AS NUMERIC(36, 2)) AS DataSize_GB,
    CAST(ROUND((CASE WHEN (a1.used + ISNULL(a4.used,0)) > a1.data THEN (a1.used + ISNULL(a4.used,0)) - a1.data ELSE 0 END) * 8 / 1024.00 / 1024.00, 2) AS NUMERIC(36, 2)) AS IndexSize_GB,
    CAST(ROUND((CASE WHEN (a1.reserved + ISNULL(a4.reserved,0)) > a1.used THEN (a1.reserved + ISNULL(a4.reserved,0)) - a1.used ELSE 0 END) * 8 / 1024.00 / 1024.00, 2) AS NUMERIC(36, 2)) AS UnusedSize_GB
FROM
    (SELECT 
        ps.object_id,
        SUM (CASE WHEN (ps.index_id < 2) THEN row_count ELSE 0 END) AS [rows],
        SUM (ps.reserved_page_count) AS reserved,
        SUM (CASE
                WHEN (ps.index_id < 2) THEN (ps.in_row_data_page_count + ps.lob_used_page_count + ps.row_overflow_used_page_count)
                ELSE (ps.lob_used_page_count + ps.row_overflow_used_page_count)
            END
            ) AS data,
        SUM (ps.used_page_count) AS used
    FROM sys.dm_db_partition_stats ps
        --===Remove the following comment for SQL Server 2014+
        --WHERE ps.object_id NOT IN (SELECT object_id FROM sys.tables WHERE is_memory_optimized = 1)
    GROUP BY ps.object_id) AS a1
LEFT OUTER JOIN 
    (SELECT 
        it.parent_id,
        SUM(ps.reserved_page_count) AS reserved,
        SUM(ps.used_page_count) AS used
     FROM sys.dm_db_partition_stats ps
     INNER JOIN sys.internal_tables it ON (it.object_id = ps.object_id)
     WHERE it.internal_type IN (202,204)
     GROUP BY it.parent_id) AS a4 ON (a4.parent_id = a1.object_id)
INNER JOIN sys.all_objects a2  ON ( a1.object_id = a2.object_id ) 
INNER JOIN sys.schemas a3 ON (a2.schema_id = a3.schema_id)
WHERE a2.type <> N'S' and a2.type <> N'IT'
--AND a2.name = 'MyTable'       --Filter for specific table
--ORDER BY a3.name, a2.name
ORDER BY ReservedSize_MB DESC
1赞 Sergey 6/22/2019 #21

下面是一个示例查询,用于获取大于 1GB 的表,按大小降序排序。

USE YourDB
GO

DECLARE @Mult float = 8
SET @Mult = @Mult / POWER(2, 20) -- Use POWER(2, 10) for MBs

; WITH CTE AS
(
SELECT
    i.object_id,
    Rows = MAX(p.rows),
    TotalSpaceGB = ROUND(SUM(a.total_pages) * @Mult, 0),
    UsedSpaceGB = ROUND(SUM(a.used_pages) * @Mult, 0)
FROM 
    sys.indexes i
JOIN
    sys.partitions p ON i.object_id = p.object_id AND i.index_id = p.index_id
JOIN
    sys.allocation_units a ON p.partition_id = a.container_id
WHERE
    i.object_id > 255
GROUP BY
    i.object_id
HAVING
    SUM(a.total_pages) * @Mult > 1
)
SELECT 
    SchemaName = s.name,
    TableName = t.name,
    c.TotalSpaceGB,
    c.UsedSpaceGB,
    UnusedSpaceGB = c.TotalSpaceGB - c.UsedSpaceGB,
    [RowCount] = c.Rows
FROM 
    CTE c
JOIN    
    sys.tables t ON t.object_id = c.object_id
JOIN
    sys.schemas s ON t.schema_id = s.schema_id
ORDER BY
    c.TotalSpaceGB DESC
46赞 Irf 7/26/2019 #22

对于 Azure,我使用了以下命令:

应具有 SSMS v17.x+

我用过;

enter image description here

有了这个,正如 Sparrow 用户所提到的

打开>并选择“表”,
然后按 F7 键,您应该会
看到:
Databasesrow countenter image description here

此处的 SSMS 已连接到 Azure 数据库

评论

0赞 Amirreza 12/19/2019
内存优化表有问题,(我看到这篇文章后刚刚进行了测试:)
3赞 Aaron Hudon 7/22/2020
右键单击列以显示其他属性,例如索引空间或使用的数据空间。
1赞 Daxesh Radadiya 8/3/2020 #23

我发现这个查询易于使用且快速。

select schema_name(tab.schema_id) + '.' + tab.name as [table], 
cast(sum(spc.used_pages * 8)/1024.00 as numeric(36, 2)) as used_mb,
cast(sum(spc.total_pages * 8)/1024.00 as numeric(36, 2)) as allocated_mb
from sys.tables (nolock) tab
inner join sys.indexes (nolock) ind 
    on tab.object_id = ind.object_id
inner join sys.partitions  (nolock) part 
    on ind.object_id = part.object_id and ind.index_id = part.index_id
inner join sys.allocation_units (nolock) spc
    on part.partition_id = spc.container_id
group by schema_name(tab.schema_id) + '.' + tab.name
order by sum(spc.used_pages) desc
1赞 K4M 8/8/2020 #24

如果您只关心数据库中的空空间浪费,而不关心单个表,则可以考虑以下几点:

如果数据库遇到大量数据插入和删除,可能就像在 ETL 情况下一样,这将导致数据库中有太多未使用的空间,因为文件组会自动增长,但永远不会自动收缩。

可以使用数据库的“属性”页查看是否属于这种情况。您可以收缩(右键单击数据库>“任务”>收缩)并收回一些空间。但是,如果根本原因仍然存在,数据库将重新增长(并消耗额外的时间尝试重新增长,并且事情会变慢,直到它增长到足够多 - 所以在这种情况下不要这样做)

(KEK: key encryption key)

32赞 Blade 9/24/2020 #25

Marc_s 的答案在使用多个分区和/或过滤索引时给出了不正确的结果。它也不区分数据和索引的大小,这通常非常相关。一些建议的修复程序并不能解决核心问题,或者也是错误的。

以下查询解决了所有这些问题。

SELECT 
     [object_id]        = t.[object_id]
    ,[schema_name]      = s.[name]
    ,[table_name]       = t.[name]
    ,[index_name]       = CASE WHEN i.[type] in (0,1,5) THEN null    ELSE i.[name] END -- 0=Heap; 1=Clustered; 5=Clustered Columnstore
    ,[object_type]      = CASE WHEN i.[type] in (0,1,5) THEN 'TABLE' ELSE 'INDEX'  END
    ,[index_type]       = i.[type_desc]
    ,[partition_count]  = p.partition_count
    ,[row_count]        = p.[rows]
    ,[data_compression] = CASE WHEN p.data_compression_cnt > 1 THEN 'Mixed'
                               ELSE (  SELECT DISTINCT p.data_compression_desc
                                       FROM sys.partitions p
                                       WHERE i.[object_id] = p.[object_id] AND i.index_id = p.index_id
                                    )
                          END
    ,[total_space_MB]   = cast(round(( au.total_pages                  * (8/1024.00)), 2) AS DECIMAL(36,2))
    ,[used_space_MB]    = cast(round(( au.used_pages                   * (8/1024.00)), 2) AS DECIMAL(36,2))
    ,[unused_space_MB]  = cast(round(((au.total_pages - au.used_pages) * (8/1024.00)), 2) AS DECIMAL(36,2))
FROM sys.schemas s
JOIN sys.tables  t ON s.schema_id = t.schema_id
JOIN sys.indexes i ON t.object_id = i.object_id
JOIN (
    SELECT [object_id], index_id, partition_count=count(*), [rows]=sum([rows]), data_compression_cnt=count(distinct [data_compression])
    FROM sys.partitions
    GROUP BY [object_id], [index_id]
) p ON i.[object_id] = p.[object_id] AND i.[index_id] = p.[index_id]
JOIN (
    SELECT p.[object_id], p.[index_id], total_pages = sum(a.total_pages), used_pages = sum(a.used_pages), data_pages=sum(a.data_pages)
    FROM sys.partitions p
    JOIN sys.allocation_units a ON p.[partition_id] = a.[container_id]
    GROUP BY p.[object_id], p.[index_id]
) au ON i.[object_id] = au.[object_id] AND i.[index_id] = au.[index_id]
WHERE t.is_ms_shipped = 0 -- Not a system table
0赞 Vinod Narwal 9/22/2021 #26
CREATE TABLE #tmp_table_info
(
id int identity(1,1),
tblname varchar(200)
);
CREATE TABLE #SpaceUsed 
(
     TableName sysname
    ,NumRows BIGINT
    ,ReservedSpace VARCHAR(50)
    ,DataSpace VARCHAR(50)
    ,IndexSize VARCHAR(50)
    ,UnusedSpace VARCHAR(50)
) 

insert into #tmp_table_info
select s.name+'.'+t.name 
from sys.tables t 
inner join sys.schemas s on t.schema_id = s.schema_id
where t.type = 'U';

declare @min int =1,@max int = 0
select @max = count(*)
from #tmp_table_info
while(@min<=@max)
begin
    declare @tablename varchar(200)
    select @tablename=tblname
    from #tmp_table_info
    where id =@min

    DECLARE @str VARCHAR(500)
    SET @str =  'sp_spaceused '''+@tablename+''''
    INSERT INTO #SpaceUsed 
    EXEC (@str)  
    set @min =@min + 1
end;
select @@SERVERNAME as servername,DB_NAME() as DatabaseName,CONVERT(numeric(18,0),REPLACE(ReservedSpace,' KB','')) / 1024 as ReservedSpace_MB,
CONVERT(numeric(18,0),REPLACE(DataSpace,' KB','')) / 1024 as DataSpace_MB,
CONVERT(numeric(18,0),REPLACE(IndexSize,' KB','')) / 1024 as IndexSpace_MB,
CONVERT(numeric(18,0),REPLACE(UnusedSpace,' KB','')) / 1024 as UnusedSpace_MB from #SpaceUsed
drop table #tmp_table_info
drop table #SpaceUsed
4赞 Mehdi Hassani Goodarzi 5/30/2022 #27

也许表格有更多的部分文件,必须显示文件顺序

SELECT
  T1.Name                                       AS TableName,
  T5.Name                                       AS SchemaName,
  T3.partition_number                           AS PartionNumber,   
  T3.Rows                                       AS RowsCount,
  SUM(T4.total_pages) * 8                       AS TotalSpaceKB,
  SUM(T4.used_pages) * 8                        AS UsedSpaceKB,
  (SUM(T4.total_pages) - SUM(T4.used_pages)) * 8 AS UnusedSpaceKB
FROM
  sys.objects T1  INNER JOIN 
  sys.indexes T2 ON T1.object_id = T2.object_id  INNER JOIN 
  sys.partitions T3 ON T2.object_id = T3.object_id AND T2.index_id = T3.index_id  INNER JOIN
  sys.allocation_units T4 ON T3.partition_id = T4.container_id LEFT JOIN
  sys.schemas T5 ON T1.schema_id = T5.schema_id
WHERE
  T1.type='U'
GROUP BY
  T1.Name, T5.Name, T3.Rows,T3.partition_number
ORDER BY
  T1.Name,T3.partition_number;
7赞 Goku - stands with Palestine 9/12/2022 #28
SELECT o.name AS ObjectName, 
       SUM(reserved_page_count) * 8.0 / 1024 AS SizeinMB
FROM sys.dm_db_partition_stats AS ps
INNER JOIN sys.sysobjects AS o
ON ps.object_id = o.id
GROUP BY o.name
ORDER BY SizeinMB DESC;