Search code examples
sqlsql-serverdatalength

size of a sql table variable without using DATALENGTH


How can I determine the space used by a table variable without using DATALENGTH on all columns?

eg:

DECLARE @T TABLE
(
a bigint,
b bigint,
c int,
d varchar(max)
)

insert into @T select 1,2,3, 'abc123'

exec sp_spaceused @T

Trying to work out how much memory a Table variable consumes when running a stored procedure.

I know in this example I can go:

SELECT DATALENGTH(a) + DATALENGTH(b) + DATALENGTH(c) + DATALENGTH(d)

But is there any other way other than doing DATALENGTH on all table columns?


Solution

  • The metadata for table variables is pretty much the same as for other types of tables so you can determine space used by looking in various system views in tempdb.

    The main obstacle is that the table variable will be given an auto generated name such as #3D7E1B63 and I'm not sure if there is a straight forward way of determining its object_id.

    The code below uses the undocumented %%physloc%% function (requires SQL Server 2008+) to determine a data page belonging to the table variable then DBCC PAGE to get the associated object_id. It then executes code copied directly from the sp_spaceused procedure to return the results.

    DECLARE @T TABLE
    (
    a bigint,
    b bigint,
    c int,
    d varchar(max)
    )
    
    insert into @T select 1,2,3, 'abc123'
    DECLARE @DynSQL nvarchar(100)
    
    SELECT TOP (1) @DynSQL = 'DBCC PAGE(2,' + 
                           CAST(file_id AS VARCHAR) + ',' + 
                           CAST(page_id AS VARCHAR) + ',1) WITH TABLERESULTS' 
    FROM @T
    CROSS APPLY sys.fn_PhysLocCracker(%%physloc%%)
    
    
    DECLARE @DBCCPage TABLE (
        [ParentObject] [varchar](100) NULL,
        [Object] [varchar](100) NULL,
        [Field] [varchar](100) NULL,
        [VALUE] [varchar](100) NULL
    ) 
    INSERT INTO @DBCCPage
    EXEC (@DynSQL)
    
    DECLARE @id int
    
    SELECT @id = VALUE
    FROM @DBCCPage 
    WHERE Field = 'Metadata: ObjectId'
    
    
    
    EXEC sp_executesql N'
    USE tempdb
    
    declare @type character(2) -- The object type.  
      ,@pages bigint   -- Working variable for size calc.  
      ,@dbname sysname  
      ,@dbsize bigint  
      ,@logsize bigint  
      ,@reservedpages  bigint  
      ,@usedpages  bigint  
      ,@rowCount bigint  
    /*  
     ** Now calculate the summary data.   
     *  Note that LOB Data and Row-overflow Data are counted as Data Pages.  
     */  
     SELECT   
      @reservedpages = SUM (reserved_page_count),  
      @usedpages = SUM (used_page_count),  
      @pages = SUM (  
       CASE  
        WHEN (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  
       ),  
      @rowCount = SUM (  
       CASE  
        WHEN (index_id < 2) THEN row_count  
        ELSE 0  
       END  
       )  
     FROM sys.dm_db_partition_stats  
     WHERE object_id = @id;  
    
     /*  
     ** Check if table has XML Indexes or Fulltext Indexes which use internal tables tied to this table  
     */  
     IF (SELECT count(*) FROM sys.internal_tables WHERE parent_id = @id AND internal_type IN (202,204,211,212,213,214,215,216)) > 0   
     BEGIN  
      /*  
      **  Now calculate the summary data. Row counts in these internal tables don''t   
      **  contribute towards row count of original table.  
      */  
      SELECT   
       @reservedpages = @reservedpages + sum(reserved_page_count),  
       @usedpages = @usedpages + sum(used_page_count)  
      FROM sys.dm_db_partition_stats p, sys.internal_tables it  
      WHERE it.parent_id = @id AND it.internal_type IN (202,204,211,212,213,214,215,216) AND p.object_id = it.object_id;  
     END  
    
     SELECT   
      name = OBJECT_NAME (@id),  
      rows = convert (char(11), @rowCount),  
      reserved = LTRIM (STR (@reservedpages * 8, 15, 0) + '' KB''),  
      data = LTRIM (STR (@pages * 8, 15, 0) + '' KB''),  
      index_size = LTRIM (STR ((CASE WHEN @usedpages > @pages THEN (@usedpages - @pages) ELSE 0 END) * 8, 15, 0) + '' KB''),  
      unused = LTRIM (STR ((CASE WHEN @reservedpages > @usedpages THEN (@reservedpages - @usedpages) ELSE 0 END) * 8, 15, 0) + '' KB'')  
    
    
    ', N'@id int',@id=@id
    

    Returns

    name                           rows        reserved           data               index_size         unused
    ------------------------------ ----------- ------------------ ------------------ ------------------ ------------------
    #451F3D2B                      1           16 KB              8 KB               8 KB               0 KB