แบบสอบถามเพื่อแสดงรายการจำนวนระเบียนในแต่ละตารางในฐานข้อมูล


199

วิธีการแสดงจำนวนแถวของแต่ละตารางในฐานข้อมูล บางคนเทียบเท่า

select count(*) from table1
select count(*) from table2
...
select count(*) from tableN

ฉันจะโพสต์โซลูชัน แต่ยินดีต้อนรับวิธีอื่น ๆ

คำตอบ:


314

หากคุณใช้ SQL Server 2005 ขึ้นไปคุณสามารถใช้สิ่งนี้:

SELECT 
    t.NAME AS TableName,
    i.name as indexName,
    p.[Rows],
    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.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
WHERE 
    t.NAME NOT LIKE 'dt%' AND
    i.OBJECT_ID > 255 AND   
    i.index_id <= 1
GROUP BY 
    t.NAME, i.object_id, i.index_id, i.name, p.[Rows]
ORDER BY 
    object_name(i.object_id) 

ในความคิดของฉันมันง่ายต่อการจัดการกว่าการsp_msforeachtableส่งออก


1
มีความคิดว่าทำไมมันถึงถูกกรองตารางด้วยชื่อที่ขึ้นต้นด้วย "dt"? ฉันเห็นสคริปต์นี้ทั่วทั้งเน็ต แต่ไม่มีคำอธิบายเกี่ยวกับเกณฑ์นั้น พวกเราทุกคนถูกหลอกหรือไม่
Skaue

6
@Skaue: หากคุณติดตั้งฟังก์ชั่น "Database Diagram" ลงในฐานข้อมูลของคุณคุณจะมีตารางที่คล้ายdtPropertiesๆ กัน เนื่องจากเป็นตาราง "ระบบ" ฉันไม่ต้องการรายงานเกี่ยวกับตารางเหล่านั้น
marc_s

1
ความเป็นไปได้ของการเติมคำนำหน้าชื่อแท็บด้วยชื่อสกีมาในสคริปต์นี้หรือไม่?
gh0st

ด้วยเหตุผลบางอย่างแบบสอบถามนี้จะไม่ส่งคืนตารางทั้งหมด ฉันมี 382 ตารางในฐานข้อมูล แต่แบบสอบถามนี้ส่งคืนแถวที่ 270 เท่านั้น (ข้อมูลตาราง) หลังจากลบเงื่อนไขที่ฉันได้รับ 302 แถว เป็นเพราะความจริงที่ว่าข้อมูลตารางบางส่วนหายไปจากหนึ่งในตาราง SYS ดังนั้นการรวมจะถูกละเว้น ฐานข้อมูลไม่มีตารางระบบใด ๆ
Ankesh Kushwah

นี่มันใช้งานได้ดี คุณสามารถแก้ไขเพื่อเปรียบเทียบสองฐานข้อมูลได้หรือไม่
sanjeewa

107

ตัวอย่างที่ฉันพบได้ที่http://www.sqlteam.com/forums/topic.asp?TOPIC_ID=21021ที่ช่วยฉัน:

select t.name TableName, i.rows Records
from sysobjects t, sysindexes i
where t.xtype = 'U' and i.id = t.id and i.indid in (0,1)
order by TableName;

5
ฉันชอบโซลูชันนี้แม้ว่าฉันจะใช้JOINไวยากรณ์from sysobjects t inner join sysindexes i on i.id = t.id and i.indid in (0,1) where t.xtype = 'U'
Shnugo

ฉันชอบที่จะใช้คำสั่ง JOIN ด้วยเช่นกัน แต่โพสต์ข้อมูลโค้ดตามที่พบ :)
Erik Anderson

32

หากต้องการรับข้อมูลนั้นใน SQL Management Studio ให้คลิกขวาที่ฐานข้อมูลจากนั้นเลือกรายงาน -> รายงานมาตรฐาน -> การใช้ดิสก์ตามตาราง


6
วิธีการประเมินต่ำกว่านี้จะสร้างรายงานที่เรียงลำดับได้อย่างรวดเร็วโดยแสดง # แถวและขนาดของข้อมูล
tbone

8
SELECT 
    T.NAME AS 'TABLE NAME',
    P.[ROWS] AS 'NO OF ROWS'
FROM SYS.TABLES T 
INNER JOIN  SYS.PARTITIONS P ON T.OBJECT_ID=P.OBJECT_ID;

3
ข้อความค้นหานี้จะส่งคืนผลลัพธ์แถวสำหรับแต่ละดัชนีในทุกตาราง เพิ่ม WHERE P.INDEX_ID IN (0,1) เพื่อ จำกัด ผลลัพธ์ส่งคืนที่ตั้งเป็นฮีปหรือดัชนีคลัสเตอร์เฉพาะที่เหมาะสมเท่านั้น
Rasmus Remmer Bielidt

6

เท่าที่เห็นที่นี่จะส่งคืนจำนวนที่ถูกต้องโดยที่เมธอดที่ใช้ตารางข้อมูลเมตาจะส่งกลับค่าประมาณการเท่านั้น

    CREATE PROCEDURE ListTableRowCounts 
    AS 
    BEGIN 
        SET NOCOUNT ON 

        CREATE TABLE #TableCounts
        ( 
            TableName VARCHAR(500), 
            CountOf INT 
        ) 

        INSERT #TableCounts
            EXEC sp_msForEachTable 
                'SELECT PARSENAME(''?'', 1), 
                COUNT(*) FROM ? WITH (NOLOCK)' 

        SELECT TableName , CountOf 
            FROM #TableCounts
            ORDER BY TableName 

        DROP TABLE #TableCounts
    END
    GO

ดังนั้นจึงดูเหมือนเป็นการประนีประนอมของการใช้ storc sp_msForEachTable ที่ไม่มีเอกสารกับการใช้ตารางระบบที่บางครั้งไม่ได้รับข้อมูลล่าสุด +1 และขอขอบคุณสำหรับการเชื่อมโยง
kristof

3
sp_MSForEachTable 'DECLARE @t AS VARCHAR(MAX); 
SELECT @t = CAST(COUNT(1) as VARCHAR(MAX)) 
+ CHAR(9) + CHAR(9) + ''?'' FROM ? ; PRINT @t'

เอาท์พุท:

ป้อนคำอธิบายรูปภาพที่นี่


ฉันต้องการบางอย่างสำหรับ SQL Server 2000 มันใช้งานได้ ขอบคุณ!
Alrekr

3

โชคดีที่สตูดิโอการจัดการ SQL Server ให้คำแนะนำในการทำเช่นนี้ ทำเช่นนี้,

  1. เริ่มการติดตาม SQL Server และเปิดกิจกรรมที่คุณกำลังทำ (กรองตาม ID การเข้าสู่ระบบของคุณหากคุณไม่ได้อยู่คนเดียวและตั้งชื่อแอปพลิเคชันเป็น Microsoft SQL Server Management Studio) หยุดการติดตามชั่วคราวและทิ้งผลลัพธ์ใด ๆ
  2. จากนั้นคลิกขวาที่ตารางแล้วเลือกคุณสมบัติจากเมนูป๊อปอัป
  3. เริ่มติดตามอีกครั้ง;
  4. ตอนนี้ในสตูดิโอการจัดการเซิร์ฟเวอร์ SQL เลือกรายการคุณสมบัติการจัดเก็บทางด้านซ้าย

หยุดการติดตามชั่วคราวและดูสิ่งที่ TSQL สร้างขึ้นโดย Microsoft

ในแบบสอบถามล่าสุดอาจคุณจะเห็นคำสั่งที่ขึ้นต้นด้วย exec sp_executesql N'SELECT

เมื่อคุณคัดลอกรหัสที่ดำเนินการไปยัง Visual Studio คุณจะสังเกตเห็นว่ารหัสนี้สร้างข้อมูลทั้งหมดที่วิศวกรที่ microsoft ใช้ในการเติมหน้าต่างคุณสมบัติ

เมื่อคุณทำการแก้ไขปานกลางกับคิวรีนั้นคุณจะได้รับสิ่งนี้:

SELECT
SCHEMA_NAME(tbl.schema_id)+'.'+tbl.name as [table], --> something I added
p.partition_number AS [PartitionNumber],
prv.value AS [RightBoundaryValue],
 fg.name AS [FileGroupName],
CAST(pf.boundary_value_on_right AS int) AS [RangeType],
CAST(p.rows AS float) AS [RowCount],
p.data_compression AS [DataCompression]
FROM sys.tables AS tbl
INNER JOIN sys.indexes AS idx ON idx.object_id = tbl.object_id and idx.index_id < 2
INNER JOIN sys.partitions AS p ON p.object_id=CAST(tbl.object_id AS int) AND p.index_id=idx.index_id
LEFT OUTER JOIN sys.destination_data_spaces AS dds ON dds.partition_scheme_id = idx.data_space_id and dds.destination_id = p.partition_number
LEFT OUTER JOIN sys.partition_schemes AS ps ON ps.data_space_id = idx.data_space_id
LEFT OUTER JOIN sys.partition_range_values AS prv ON prv.boundary_id = p.partition_number and prv.function_id = ps.function_id
LEFT OUTER JOIN sys.filegroups AS fg ON fg.data_space_id = dds.data_space_id or fg.data_space_id = idx.data_space_id
LEFT OUTER JOIN sys.partition_functions AS pf ON  pf.function_id = prv.function_id

ตอนนี้แบบสอบถามไม่สมบูรณ์และคุณสามารถอัปเดตให้ตรงกับคำถามอื่น ๆ ที่คุณอาจมีประเด็นก็คือคุณสามารถใช้ความรู้ของ microsoft เพื่อตอบคำถามส่วนใหญ่ที่คุณมีโดยดำเนินการข้อมูลที่คุณสนใจและติดตาม TSQL ที่สร้างโดยใช้ profiler

ฉันชอบที่จะคิดว่าวิศวกรของ MS รู้วิธีการทำงานของเซิร์ฟเวอร์ SQL และมันจะสร้าง TSQL ที่ทำงานกับทุกรายการที่คุณสามารถทำงานกับการใช้เวอร์ชันบน SSMS ที่คุณใช้อยู่ดังนั้นมันจึงค่อนข้างดีสำหรับรุ่นที่มีขนาดใหญ่ในปัจจุบัน อนาคต.

และจำไว้ว่าอย่าเพิ่งลอกเลียนแบบลองทำความเข้าใจกับสิ่งอื่นที่คุณอาจพบด้วยวิธีที่ผิด

วอลเตอร์


2

วิธีการนี้ใช้การต่อเรียงสตริงเพื่อสร้างคำสั่งที่มีตารางทั้งหมดและการนับแบบไดนามิกเช่นตัวอย่างที่ให้ไว้ในคำถามเดิม:

          SELECT COUNT(*) AS Count,'[dbo].[tbl1]' AS TableName FROM [dbo].[tbl1]
UNION ALL SELECT COUNT(*) AS Count,'[dbo].[tbl2]' AS TableName FROM [dbo].[tbl2]
UNION ALL SELECT...

ในที่สุดนี้จะดำเนินการกับEXEC:

DECLARE @cmd VARCHAR(MAX)=STUFF(
                    (
                        SELECT 'UNION ALL SELECT COUNT(*) AS Count,''' 
                              + QUOTENAME(t.TABLE_SCHEMA) + '.' + QUOTENAME(t.TABLE_NAME) 
                              + ''' AS TableName FROM ' + QUOTENAME(t.TABLE_SCHEMA) + '.' + QUOTENAME(t.TABLE_NAME)
                        FROM INFORMATION_SCHEMA.TABLES AS t
                        WHERE TABLE_TYPE='BASE TABLE'
                        FOR XML PATH('')
                    ),1,10,'');
EXEC(@cmd);

โปรดทราบว่าการแก้ปัญหานี้รวมถึงชื่อของสคี (ซึ่งอาจเป็นประโยชน์)
gordon613

2

วิธีที่เร็วที่สุดในการค้นหาจำนวนแถวของตารางทั้งหมดใน SQL Refreence ( http://www.codeproject.com/Tips/811017/Fastest-way-to-find-row-count-of-all-tables-in- SQL )

SELECT T.name AS [TABLE NAME], I.rows AS [ROWCOUNT] 
    FROM   sys.tables AS T 
       INNER JOIN sys.sysindexes AS I ON T.object_id = I.id 
       AND I.indid < 2 
ORDER  BY I.rows DESC

1

สิ่งแรกที่นึกถึงคือการใช้ sp_msForEachTable

exec sp_msforeachtable 'select count(*) from ?'

ที่ไม่ได้แสดงรายการชื่อตารางดังนั้นจึงสามารถขยายได้

exec sp_msforeachtable 'select parsename(''?'', 1),  count(*) from ?'

ปัญหาที่นี่คือถ้าฐานข้อมูลมีมากกว่า 100 ตารางคุณจะได้รับข้อความแสดงข้อผิดพลาดต่อไปนี้:

แบบสอบถามมีจำนวนชุดผลลัพธ์เกินจำนวนสูงสุดที่สามารถแสดงได้ในตารางผลลัพธ์ เฉพาะชุดผลลัพธ์ 100 ชุดแรกเท่านั้นที่จะแสดงในตาราง

ดังนั้นฉันจึงลงเอยด้วยการใช้ตัวแปรตารางเพื่อเก็บผลลัพธ์

declare @stats table (n sysname, c int)
insert into @stats
    exec sp_msforeachtable 'select parsename(''?'', 1),  count(*) from ?'
select 
    * 
from @stats
order by c desc

1

คำตอบที่ได้รับการยอมรับไม่ได้ทำงานสำหรับฉันใน SQL Azure ที่นี่เป็นหนึ่งที่ได้ก็สุดได้อย่างรวดเร็วและไม่ตรงกับสิ่งที่ฉันต้องการ:

select t.name, s.row_count
from sys.tables t
join sys.dm_db_partition_stats s
  ON t.object_id = s.object_id
    and t.type_desc = 'USER_TABLE'
    and t.name not like '%dss%'
    and s.index_id = 1
order by s.row_count desc

1

สคริปต์ sql นี้ให้สคีมาชื่อตารางและจำนวนแถวของแต่ละตารางในฐานข้อมูลที่เลือก:

SELECT SCHEMA_NAME(schema_id) AS [SchemaName],
[Tables].name AS [TableName],
SUM([Partitions].[rows]) AS [TotalRowCount]
FROM sys.tables AS [Tables]
JOIN sys.partitions AS [Partitions]
ON [Tables].[object_id] = [Partitions].[object_id]
AND [Partitions].index_id IN ( 0, 1 )
-- WHERE [Tables].name = N'name of the table'
GROUP BY SCHEMA_NAME(schema_id), [Tables].name
order by [TotalRowCount] desc

Ref: https://blog.sqlauthority.com/2017/05/24/sql-server-find-row-count-every-every-table-database-efficiently/

อีกวิธีในการทำสิ่งนี้:

SELECT  o.NAME TABLENAME,
  i.rowcnt 
FROM sysindexes AS i
  INNER JOIN sysobjects AS o ON i.id = o.id 
WHERE i.indid < 2  AND OBJECTPROPERTY(o.id, 'IsMSShipped') = 0
ORDER BY i.rowcnt desc

0

ฉันคิดว่าวิธีที่สั้นที่สุดเร็วที่สุดและง่ายที่สุดคือ:

SELECT
    object_name(object_id) AS [Table],
    SUM(row_count) AS [Count]
FROM
    sys.dm_db_partition_stats
WHERE
    --object_schema_name(object_id) = 'dbo' AND 
    index_id < 2
GROUP BY
    object_id

0

คุณสามารถลองสิ่งนี้:

SELECT  OBJECT_SCHEMA_NAME(ps.object_Id) AS [schemaname],
        OBJECT_NAME(ps.object_id) AS [tablename],
        row_count AS [rows]
FROM sys.dm_db_partition_stats ps
WHERE OBJECT_SCHEMA_NAME(ps.object_Id) <> 'sys' AND ps.index_id < 2
ORDER BY 
        OBJECT_SCHEMA_NAME(ps.object_Id),
        OBJECT_NAME(ps.object_id)

0
USE DatabaseName
CREATE TABLE #counts
(
    table_name varchar(255),
    row_count int
)

EXEC sp_MSForEachTable @command1='INSERT #counts (table_name, row_count) SELECT ''?'', COUNT(*) FROM ?'
SELECT table_name, row_count FROM #counts ORDER BY table_name, row_count DESC
DROP TABLE #counts

0

จากคำถามนี้: /dba/114958/list-all-tables-from-all-user-database/230411#230411

ฉันเพิ่มจำนวนระเบียนลงในคำตอบที่ @Aaron Bertrand ให้ไว้ซึ่งแสดงรายการฐานข้อมูลทั้งหมดและตารางทั้งหมด

DECLARE @src NVARCHAR(MAX), @sql NVARCHAR(MAX);

SELECT @sql = N'', @src = N' UNION ALL 
SELECT ''$d'' as ''database'', 
    s.name COLLATE SQL_Latin1_General_CP1_CI_AI as ''schema'',
    t.name COLLATE SQL_Latin1_General_CP1_CI_AI as ''table'' ,
    ind.rows as record_count
  FROM [$d].sys.schemas AS s
  INNER JOIN [$d].sys.tables AS t ON s.[schema_id] = t.[schema_id]
  INNER JOIN [$d].sys.sysindexes AS ind ON t.[object_id] = ind.[id]
  where ind.indid < 2';

SELECT @sql = @sql + REPLACE(@src, '$d', name)
  FROM sys.databases
  WHERE database_id > 4
    AND [state] = 0
    AND HAS_DBACCESS(name) = 1;

SET @sql = STUFF(@sql, 1, 10, CHAR(13) + CHAR(10));

PRINT @sql;
--EXEC sys.sp_executesql @sql;

0

คุณสามารถคัดลอกวางและเรียกใช้โค้ดนี้เพื่อรับจำนวนระเบียนทั้งหมดในตาราง หมายเหตุ: รหัสถูกใส่ความคิดเห็นพร้อมคำแนะนำ

create procedure RowCountsPro
as
begin
--drop the table if exist on each exicution
IF OBJECT_ID (N'dbo.RowCounts', N'U') IS NOT NULL 
DROP TABLE dbo.RowCounts;
-- creating new table
CREATE TABLE RowCounts 
( [TableName]            VARCHAR(150)
, [RowCount]               INT
, [Reserved]                 NVARCHAR(50)
, [Data]                        NVARCHAR(50)
, [Index_Size]               NVARCHAR(50)
, [UnUsed]                   NVARCHAR(50))
--inserting all records
INSERT INTO RowCounts([TableName], [RowCount],[Reserved],[Data],[Index_Size],[UnUsed])
--  "sp_MSforeachtable" System Procedure, 'sp_spaceused "?"' param to get records and resources used
EXEC sp_MSforeachtable 'sp_spaceused "?"' 
-- selecting data and returning a table of data
SELECT [TableName], [RowCount],[Reserved],[Data],[Index_Size],[UnUsed]
FROM RowCounts
ORDER BY [TableName]
end

ฉันทดสอบโค้ดนี้แล้วและใช้งานได้ดีกับ SQL Server 2014


0

ฉันต้องการแบ่งปันสิ่งที่ทำงานให้ฉัน

SELECT
      QUOTENAME(SCHEMA_NAME(sOBJ.schema_id)) + '.' + QUOTENAME(sOBJ.name) AS [TableName]
      , SUM(sdmvPTNS.row_count) AS [RowCount]
FROM
      sys.objects AS sOBJ
      INNER JOIN sys.dm_db_partition_stats AS sdmvPTNS
            ON sOBJ.object_id = sdmvPTNS.object_id
WHERE 
      sOBJ.type = 'U'
      AND sOBJ.is_ms_shipped = 0x0
      AND sdmvPTNS.index_id < 2
GROUP BY
      sOBJ.schema_id
      , sOBJ.name
ORDER BY [TableName]
GO

ฐานข้อมูลโฮสต์ใน Azure และผลลัพธ์สุดท้ายคือ: ป้อนคำอธิบายรูปภาพที่นี่

เครดิต: https://www.mssqltips.com/sqlservertip/2537/sql-server-row-count-for-all-tables-in-a-database/


-1

ถ้าคุณใช้ MySQL> 4.x คุณสามารถใช้สิ่งนี้:

select TABLE_NAME, TABLE_ROWS from information_schema.TABLES where TABLE_SCHEMA="test";

โปรดทราบว่าสำหรับเอ็นจิ้นการเก็บข้อมูลบางอย่าง TABLE_ROWS เป็นการประมาณ


6
เขากล่าวถึง "SQL เซิร์ฟเวอร์" ในโพสต์ของเขา (ในขณะที่แท็ก) ซึ่งเป็น Microsoft SQL Server
marc_s

-1
select T.object_id, T.name, I.indid, I.rows 
  from Sys.tables T 
  left join Sys.sysindexes I 
    on (I.id = T.object_id and (indid =1 or indid =0 ))
 where T.type='U'

นี่indid=1หมายถึงดัชนีกลุ่มและindid=0เป็นกอง


4
สวัสดีและยินดีต้อนรับสู่ Stack Overflow คำตอบนี้เหมือนกับคำตอบที่มีอายุหนึ่งปีแล้ว ... ไม่จำเป็นต้องโพสต์อีกครั้ง
Ben
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.