SQL Server의 테이블 및 인덱스 크기


90

기본적으로 SQl Server에서 테이블 및 인덱스 크기를 보는 데 도움이되는 SQL 쿼리를 가질 수 있습니까?

SQL Server는 테이블 / 인덱스에 대한 메모리 사용량을 어떻게 유지합니까?


1
또한 저장 프로 시저를 찾을 수 sp_helpdb유용
잭 버트

1
이 이미 답변,하지만 난 개인적으로이 링크 쿼리 사용됩니다 qualityofdata.com/2011/02/02/...
naiem

답변:


73

exec sp_spaceused매개 변수없이 전체 데이터베이스에 대한 요약 정보를 보여줍니다. foreachtable 솔루션은 테이블 당 하나의 결과 집합을 생성합니다. 테이블이 너무 많으면 SSMS가 처리하지 못할 수 있습니다.

저는 테이블 정보를 수집하고 크기별로 정렬 된 단일 레코드 세트에 요약을 표시 하는 스크립트 를 만들었습니다 sp_spaceused.

create table #t
(
  name nvarchar(128),
  rows varchar(50),
  reserved varchar(50),
  data varchar(50),
  index_size varchar(50),
  unused varchar(50)
)

declare @id nvarchar(128)
declare c cursor for
select '[' + sc.name + '].[' + s.name + ']' FROM sysobjects s INNER JOIN sys.schemas sc ON s.uid = sc.schema_id where s.xtype='U'

open c
fetch c into @id

while @@fetch_status = 0 begin

  insert into #t
  exec sp_spaceused @id

  fetch c into @id
end

close c
deallocate c

select * from #t
order by convert(int, substring(data, 1, len(data)-3)) desc

drop table #t

4
스크립트는 스키마 'dbo'의 테이블 만 처리합니다. 데이터베이스에 'Audit'스키마가있는 테이블이있는 경우 sp_spaceused는 exec sp_spaceused 'Audit.Data'와 같이 호출해야합니다. 따라서 스크립트는 다른 스키마의 테이블에 대한 데이터를 반환하기 위해 스키마 이름 (점으로 구분)으로 시작하는 테이블 이름을 제공하도록 수정해야합니다.
Baodad

1
@Boadad의 좋은 점 ... 그것은 매우 쉬운 수정이어야합니다. "select name from sysobjects where xtype = 'U'"는 트릭을 수행해야합니다. "select '['+ sc.name + ']. ['+ s.name + ']'FROM sysobjects s ​​INNER JOIN sys .schemas sc ON s.uid = sc.schema_id where s.xtype = 'U' "훌륭한 스크립트입니다. 감사합니다!
DCaugs

임시 테이블을 사용하는 대신 임시가 아닌 다른 테이블에 데이터를 삽입 할 수 있습니까?
Prabhakar

@PrabhakarPandey 물론 #.
Racer SQL

120

sp_spaceused는 결합 된 모든 인덱스의 크기를 제공합니다.

테이블에 대한 각 인덱스의 크기를 원하는 경우 다음 두 쿼리 중 하나를 사용하십시오.

SELECT
    i.name                  AS IndexName,
    SUM(s.used_page_count) * 8   AS IndexSizeKB
FROM sys.dm_db_partition_stats  AS s 
JOIN sys.indexes                AS i
ON s.[object_id] = i.[object_id] AND s.index_id = i.index_id
WHERE s.[object_id] = object_id('dbo.TableName')
GROUP BY i.name
ORDER BY i.name

SELECT
    i.name              AS IndexName,
    SUM(page_count * 8) AS IndexSizeKB
FROM sys.dm_db_index_physical_stats(
    db_id(), object_id('dbo.TableName'), NULL, NULL, 'DETAILED') AS s
JOIN sys.indexes AS i
ON s.[object_id] = i.[object_id] AND s.index_id = i.index_id
GROUP BY i.name
ORDER BY i.name

결과는 일반적으로 약간 다르지만 1 % 이내입니다.


첫 번째 쿼리에는 몇 가지 이유로 약간 혼란스러운 기본 키가 포함되어 있습니다.
quillbreaker

두 번째 쿼리가 Msg 102, Level 15, State 1, Line 5 - Incorrect syntax near '('.나에게 던졌지 만 구문에 문제가 없습니다. 어떤 생각?
Oliver

Oliver, 어떤 버전에서 실행 중입니까? 2008R2 및 2012에서 그대로 작동합니다.
Rob Garrison

24

SQL 2012에서는 테이블 수준에서이 정보를 얻는 것이 매우 간단 해졌습니다.

SQL Management Studio-> Db-> Reports-> Standard Reports-> Disk usage by table!

즐겨


13
EXEC sp_MSforeachtable @command1="EXEC sp_spaceused '?'"

3
당신이 포스트 코드, XML 또는 데이터 샘플, 경우 하시기 바랍니다 텍스트 편집기에서 그 라인을 강조하고 "코드 샘플"버튼 (클릭 { }친절에 편집기 도구 모음) 형식 및 구문을 강조!
marc_s

4
--Gets the size of each index for the specified table
DECLARE @TableName sysname = N'SomeTable';

SELECT i.name AS IndexName
      ,8 * SUM(s.used_page_count) AS IndexSizeKB
FROM sys.indexes AS i
    INNER JOIN sys.dm_db_partition_stats AS s 
        ON i.[object_id] = s.[object_id] AND i.index_id = s.index_id
WHERE s.[object_id] = OBJECT_ID(@TableName, N'U')
GROUP BY i.name
ORDER BY i.name;

SELECT i.name AS IndexName
      ,8 * SUM(a.used_pages) AS IndexSizeKB
FROM sys.indexes AS i
    INNER JOIN sys.partitions AS p 
        ON i.[object_id]  = p.[object_id] AND i.index_id = p.index_id
    INNER JOIN sys.allocation_units AS a 
        ON p.partition_id = a.container_id
WHERE i.[object_id] = OBJECT_ID(@TableName, N'U')
GROUP BY i.name
ORDER BY i.name;

3

다음은 가장 성공적인 답변의 더 간결한 버전입니다.

create table #tbl(
  name nvarchar(128),
  rows varchar(50),
  reserved varchar(50),
  data varchar(50),
  index_size varchar(50),
  unused varchar(50)
)

exec sp_msforeachtable 'insert into #tbl exec sp_spaceused [?]'

select * from #tbl
    order by convert(int, substring(data, 1, len(data)-3)) desc

drop table #tbl

3

이 게시물을 작성한 지 오래되었지만 스크립트를 공유하고 싶었습니다.

WITH CteIndex
AS
(
SELECT 
     reservedpages = (reserved_page_count)
     ,usedpages = (used_page_count)
     ,pages = (
            CASE
                WHEN (s.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
            )    
     ,s.object_id   
     ,i.index_id        
     ,i.type_desc AS IndexType
     ,i.name AS indexname
    FROM sys.dm_db_partition_stats s
    INNER JOIN sys.indexes i ON s.[object_id] = i.[object_id] AND s.index_id = i.index_id   
)
SELECT DISTINCT
DB_NAME(DB_ID()) AS DatabaseName
,o.name AS TableName
,o.object_id
,ct.indexname
,ct.IndexType
,ct.index_id
, IndexSpace = LTRIM (STR ((CASE WHEN usedpages > pages THEN CASE WHEN ct.index_id < 2 THEN  pages ELSE (usedpages - pages) END ELSE 0 END) * 8, 15, 0) + ' KB')
FROM CteIndex ct
INNER JOIN sys.objects o ON o.object_id = ct.object_id
INNER JOIN sys.dm_db_index_physical_stats (DB_ID(), NULL, NULL, NULL , NULL) ps ON ps.object_id = o.object_id
AND ps.index_id = ct.index_id
ORDER BY name ASC

그것은 작동합니다 :

  • SQL Server (2008부터)
  • 현재 데이터베이스 당 모든 테이블에 대한 정보를 포함합니다.

0

sp_spaceused이 정보를 가져 오는 확장 저장 프로 시저 가 있습니다. 데이터 사전에서 수행하는 것은 상당히 복잡하지만 이 링크는이 를 수행하는 스크립트로 확장됩니다. 이 스택 오버플로 질문 에는 용량 계획을위한 테이블 및 인덱스 크기의 추정치를 구성하는 데 사용할 수있는 기본 데이터 구조에 대한 정보에 대한 팬 아웃이 있습니다.


0

이 쿼리는 두 가지 다른 답변에서 나옵니다.

데이터베이스의 모든 테이블 크기 가져 오기

SQL Server 데이터베이스에서 가장 큰 개체를 찾는 방법은 무엇입니까?

, 그러나 나는 이것을 보편적으로 강화했습니다. sys.objects사전을 사용 합니다.

SELECT 
    s.NAME as SCHEMA_NAME,
    t.NAME AS OBJ_NAME,
    t.type_desc as OBJ_TYPE,
    i.name as indexName,
    sum(p.rows) as RowCounts,
    sum(a.total_pages) as TotalPages, 
    sum(a.used_pages) as UsedPages, 
    sum(a.data_pages) as DataPages,
    (sum(a.total_pages) * 8) / 1024 as TotalSpaceMB, 
    (sum(a.used_pages) * 8) / 1024 as UsedSpaceMB, 
    (sum(a.data_pages) * 8) / 1024 as DataSpaceMB
FROM 
    sys.objects t
INNER JOIN
    sys.schemas s ON t.SCHEMA_ID = s.SCHEMA_ID 
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
WHERE 
    t.NAME NOT LIKE 'dt%' AND
    i.OBJECT_ID > 255 AND   
    i.index_id <= 1
GROUP BY 
    s.NAME, t.NAME, t.type_desc, i.object_id, i.index_id, i.name 
ORDER BY
    sum(a.total_pages) DESC
;
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.