SQL Server: Filter output of sp_who2

后端 未结 13 2076
长情又很酷
长情又很酷 2020-12-02 03:49

Under SQL Server, is there an easy way to filter the output of sp_who2? Say I wanted to just show rows for a certain database, for example.

相关标签:
13条回答
  • 2020-12-02 04:19

    You could save the results into a temp table, but it would be even better to go directly to the source on master.dbo.sysprocesses.

    Here's a query that will return almost the exact same result as sp_who2:

    SELECT  spid,
            sp.[status],
            loginame [Login],
            hostname, 
            blocked BlkBy,
            sd.name DBName, 
            cmd Command,
            cpu CPUTime,
            physical_io DiskIO,
            last_batch LastBatch,
            [program_name] ProgramName   
    FROM master.dbo.sysprocesses sp 
    JOIN master.dbo.sysdatabases sd ON sp.dbid = sd.dbid
    ORDER BY spid 
    

    Now you can easily add any ORDER BY or WHERE clauses you like to get meaningful output.


    Alternatively, you might consider using Activity Monitor in SSMS (Ctrl + Alt + A) as well

    0 讨论(0)
  • 2020-12-02 04:22

    A really easy way to do it is to create an ODBC link in EXCEL and run SP_WHO2 from there.

    You can Refresh whenever you like and because it's EXCEL everything can be manipulated easily!

    0 讨论(0)
  • 2020-12-02 04:27

    One way is to create a temp table:

    CREATE TABLE #sp_who2 
    (
       SPID INT,  
       Status VARCHAR(1000) NULL,  
       Login SYSNAME NULL,  
       HostName SYSNAME NULL,  
       BlkBy SYSNAME NULL,  
       DBName SYSNAME NULL,  
       Command VARCHAR(1000) NULL,  
       CPUTime INT NULL,  
       DiskIO INT NULL,  
       LastBatch VARCHAR(1000) NULL,  
       ProgramName VARCHAR(1000) NULL,  
       SPID2 INT
    ) 
    GO
    
    INSERT INTO #sp_who2
    EXEC sp_who2
    GO
    
    SELECT *
    FROM #sp_who2
    WHERE Login = 'bla'
    GO
    
    DROP TABLE #sp_who2
    GO
    
    0 讨论(0)
  • 2020-12-02 04:28

    based on http://web.archive.org/web/20080218124946/http://sqlserver2005.databases.aspfaq.com/how-do-i-mimic-sp-who2.html

    i have created following script ,
    which resolves finding active connections to any datbase using DMV this works under sql 2005 , 2008 and 2008R2

    Following script uses sys.dm_exec_sessions , sys.dm_exec_requests , sys.dm_exec_connections , sys.dm_tran_locks

    Declare @dbName varchar(1000)
    set @dbName='abc'
    
    ;WITH DBConn(SPID,[Status],[Login],HostName,DBName,Command,LastBatch,ProgramName)
    As
    (
    SELECT 
        SPID = s.session_id,
        Status = UPPER(COALESCE
            (
                r.status,
                ot.task_state,
                s.status, 
            '')),
        [Login] = s.login_name,
        HostName = COALESCE
            (
                s.[host_name],
                '  .'
            ),
        DBName = COALESCE
            (
                DB_NAME(COALESCE
                (
                    r.database_id,
                    t.database_id
                )),
                ''
            ),
        Command = COALESCE
            (
                r.Command,
                r.wait_type,
                wt.wait_type,
                r.last_wait_type,
                ''
            ),
        LastBatch = COALESCE
            (
                r.start_time,
                s.last_request_start_time
            ),
        ProgramName = COALESCE
            (
                s.program_name, 
                ''
            )
    FROM
        sys.dm_exec_sessions s
    LEFT OUTER JOIN
        sys.dm_exec_requests r
    ON
        s.session_id = r.session_id
    LEFT OUTER JOIN
        sys.dm_exec_connections c
    ON
        s.session_id = c.session_id
    LEFT OUTER JOIN
    (
        SELECT 
            request_session_id,
            database_id = MAX(resource_database_id)
        FROM
            sys.dm_tran_locks
        GROUP BY
            request_session_id
    ) t
    ON
        s.session_id = t.request_session_id
    LEFT OUTER JOIN
        sys.dm_os_waiting_tasks wt
    ON 
        s.session_id = wt.session_id
    LEFT OUTER JOIN
        sys.dm_os_tasks ot
    ON 
        s.session_id = ot.session_id
    LEFT OUTER JOIN
    (
        SELECT
            ot.session_id,
            CPU_Time = MAX(usermode_time)
        FROM
            sys.dm_os_tasks ot
        INNER JOIN
            sys.dm_os_workers ow
        ON
            ot.worker_address = ow.worker_address
        INNER JOIN
            sys.dm_os_threads oth
        ON
            ow.thread_address = oth.thread_address
        GROUP BY
            ot.session_id
    ) tt
    ON
        s.session_id = tt.session_id
    WHERE
        COALESCE
        (
            r.command,
            r.wait_type,
            wt.wait_type,
            r.last_wait_type,
            'a'
        ) >= COALESCE
        (
            '', 
            'a'
        )
    )
    
    Select * from DBConn
    where DBName like '%'+@dbName+'%'
    
    0 讨论(0)
  • 2020-12-02 04:30

    Slight improvement to Astander's answer. I like to put my criteria at top, and make it easier to reuse day to day:

    DECLARE @Spid INT, @Status VARCHAR(MAX), @Login VARCHAR(MAX), @HostName VARCHAR(MAX), @BlkBy VARCHAR(MAX), @DBName VARCHAR(MAX), @Command VARCHAR(MAX), @CPUTime INT, @DiskIO INT, @LastBatch VARCHAR(MAX), @ProgramName VARCHAR(MAX), @SPID_1 INT, @REQUESTID INT
    
        --SET @SPID = 10
        --SET @Status = 'BACKGROUND'
        --SET @LOGIN = 'sa'
        --SET @HostName = 'MSSQL-1'
        --SET @BlkBy = 0
        --SET @DBName = 'master'
        --SET @Command = 'SELECT INTO'
        --SET @CPUTime = 1000
        --SET @DiskIO = 1000
        --SET @LastBatch = '10/24 10:00:00'
        --SET @ProgramName = 'Microsoft SQL Server Management Studio - Query'
        --SET @SPID_1 = 10
        --SET @REQUESTID = 0
    
        SET NOCOUNT ON 
        DECLARE @Table TABLE(
                SPID INT,
                Status VARCHAR(MAX),
                LOGIN VARCHAR(MAX),
                HostName VARCHAR(MAX),
                BlkBy VARCHAR(MAX),
                DBName VARCHAR(MAX),
                Command VARCHAR(MAX),
                CPUTime INT,
                DiskIO INT,
                LastBatch VARCHAR(MAX),
                ProgramName VARCHAR(MAX),
                SPID_1 INT,
                REQUESTID INT
        )
        INSERT INTO @Table EXEC sp_who2
        SET NOCOUNT OFF
        SELECT  *
        FROM    @Table
        WHERE
        (@Spid IS NULL OR SPID = @Spid)
        AND (@Status IS NULL OR Status = @Status)
        AND (@Login IS NULL OR Login = @Login)
        AND (@HostName IS NULL OR HostName = @HostName)
        AND (@BlkBy IS NULL OR BlkBy = @BlkBy)
        AND (@DBName IS NULL OR DBName = @DBName)
        AND (@Command IS NULL OR Command = @Command)
        AND (@CPUTime IS NULL OR CPUTime >= @CPUTime)
        AND (@DiskIO IS NULL OR DiskIO >= @DiskIO)
        AND (@LastBatch IS NULL OR LastBatch >= @LastBatch)
        AND (@ProgramName IS NULL OR ProgramName = @ProgramName)
        AND (@SPID_1 IS NULL OR SPID_1 = @SPID_1)
        AND (@REQUESTID IS NULL OR REQUESTID = @REQUESTID)
    
    0 讨论(0)
  • 2020-12-02 04:32

    Extension of the first and best answer... I have created a stored procedure on the master database that you can then pass parameters to .. such as the name of the database:

    USE master
    GO
    
    CREATE PROCEDURE sp_who_db
    (
        @sDBName varchar(200)   = null,
        @sStatus varchar(200)   = null,
        @sCommand varchar(200)  = null,
        @nCPUTime int           = null
    )
    AS
    DECLARE @Table TABLE
    (
        SPID INT,
        Status VARCHAR(MAX),
        LOGIN VARCHAR(MAX),
        HostName VARCHAR(MAX),
        BlkBy VARCHAR(MAX),
        DBName VARCHAR(MAX),
        Command VARCHAR(MAX),
        CPUTime INT,
        DiskIO INT,
        LastBatch VARCHAR(MAX),
        ProgramName VARCHAR(MAX),
        SPID_1 INT,
        REQUESTID INT
    )
    
    INSERT INTO @Table EXEC sp_who2
    
    SELECT  *
        FROM    @Table
        WHERE   (@sDBName IS NULL OR DBName = @sDBName)
        AND     (@sStatus IS NULL OR Status = @sStatus)
        AND     (@sCommand IS NULL OR Command = @sCommand)
        AND     (@nCPUTime IS NULL OR CPUTime > @nCPUTime)
    GO 
    

    I might extend it to add an order by parameter or even a kill paramatmer so it kills all connections to a particular data

    0 讨论(0)
提交回复
热议问题