ค้นหาสตริงโดยค้นหาตารางทั้งหมดใน SQL Server Management Studio 2008


207

มีวิธีการค้นหาสตริงในตารางฐานข้อมูลทั้งหมดใน SQL Server Management Studio 2008 หรือไม่

johnฉันต้องการที่จะค้นหาสตริงพูด johnผลที่ควรจะแสดงตารางและแถวของตนที่มี




phpmyadmin อนุญาตสิ่งนี้อย่างง่ายดาย
Matoeil

คำตอบ:


208

หากคุณเป็นเหมือนฉันและมีข้อ จำกัด บางอย่างในสภาพแวดล้อมการผลิตคุณอาจต้องการใช้ตัวแปรตารางแทนตาราง temp และแบบสอบถาม ad-hoc แทนขั้นตอนการสร้าง

แน่นอนขึ้นอยู่กับอินสแตนซ์ของเซิร์ฟเวอร์ sql นั้นจะต้องสนับสนุนตัวแปรตาราง

ฉันยังเพิ่มคำสั่ง USE เพื่อ จำกัด ขอบเขตการค้นหา

USE DATABASE_NAME
DECLARE @SearchStr nvarchar(100) = 'SEARCH_TEXT'
DECLARE @Results TABLE (ColumnName nvarchar(370), ColumnValue nvarchar(3630))

SET NOCOUNT ON

DECLARE @TableName nvarchar(256), @ColumnName nvarchar(128), @SearchStr2 nvarchar(110)
SET  @TableName = ''
SET @SearchStr2 = QUOTENAME('%' + @SearchStr + '%','''')

WHILE @TableName IS NOT NULL

BEGIN
    SET @ColumnName = ''
    SET @TableName = 
    (
        SELECT MIN(QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME))
        FROM     INFORMATION_SCHEMA.TABLES
        WHERE         TABLE_TYPE = 'BASE TABLE'
            AND    QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME) > @TableName
            AND    OBJECTPROPERTY(
                    OBJECT_ID(
                        QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME)
                         ), 'IsMSShipped'
                           ) = 0
    )

    WHILE (@TableName IS NOT NULL) AND (@ColumnName IS NOT NULL)

    BEGIN
        SET @ColumnName =
        (
            SELECT MIN(QUOTENAME(COLUMN_NAME))
            FROM     INFORMATION_SCHEMA.COLUMNS
            WHERE         TABLE_SCHEMA    = PARSENAME(@TableName, 2)
                AND    TABLE_NAME    = PARSENAME(@TableName, 1)
                AND    DATA_TYPE IN ('char', 'varchar', 'nchar', 'nvarchar', 'int', 'decimal')
                AND    QUOTENAME(COLUMN_NAME) > @ColumnName
        )

        IF @ColumnName IS NOT NULL

        BEGIN
            INSERT INTO @Results
            EXEC
            (
                'SELECT ''' + @TableName + '.' + @ColumnName + ''', LEFT(' + @ColumnName + ', 3630) 
                FROM ' + @TableName + ' (NOLOCK) ' +
                ' WHERE ' + @ColumnName + ' LIKE ' + @SearchStr2
            )
        END
    END    
END

SELECT ColumnName, ColumnValue FROM @Results

14
น่ากลัว ฉันเพิ่ม "PRINT @TableName + '.' + @ColumnName "ที่บรรทัด 53 ดังนั้นฉันจะได้รับสถานะเนื่องจากทำงานผ่านฐานข้อมูลขนาดใหญ่ของฉัน แต่สิ่งนี้มีประโยชน์มาก
ProVega

ขอบคุณ! ทำให้การค้นหา ACT เก่า! ฐานข้อมูล CRM ง่ายขึ้นโดยที่หน้าจอ fieldon เป็น "บริษัท reg" และฟิลด์ฐานข้อมูลคือ "tickersymbol" อย่างไรก็ตามสิ่งที่เกิดขึ้น! lol
Liam Wheldon

ฉันจะดึงข้อมูลคีย์หลักด้วยผลลัพธ์ได้อย่างไร ความช่วยเหลือใด ๆ
Dupinder Singh

50

สายไปหน่อย แต่หวังว่าจะเป็นประโยชน์

ทำไมไม่ลองใช้เครื่องมือของบุคคลที่สามที่สามารถรวมเข้ากับ SSMS ได้

ฉันได้ทำงานกับApexSQL Search (ฟรี 100%) ที่ประสบความสำเร็จดีทั้งกับ schema และการค้นหาข้อมูลและยังมีชุดเครื่องมือ SSMSที่มีคุณสมบัตินี้ (ไม่ฟรีสำหรับ SQL 2012 แต่ราคาไม่แพงมาก)

ขั้นตอนการจัดเก็บด้านบนนั้นยอดเยี่ยมมาก เป็นเพียงความสะดวกในความคิดของฉัน นอกจากนี้ยังต้องการการแก้ไขเล็กน้อยหากคุณต้องการค้นหาคอลัมน์วันที่และเวลาหรือคอลัมน์ GUID และ ...


34

หากต้องการอัปเดตคำตอบของTechDoสำหรับ SQL Server 2012 คุณต้องเปลี่ยน: 'FROM ' + @TableName + ' (NOLOCK) ' เป็น FROM ' + @TableName + 'WITH (NOLOCK) ' +

ฉลาดอื่น ๆ คุณจะได้รับข้อผิดพลาดต่อไปนี้: Deprecated feature 'Table hint without WITH' is not supported in this version of SQL Server.

ด้านล่างเป็นขั้นตอนการจัดเก็บที่อัปเดตอย่างสมบูรณ์:

CREATE PROC SearchAllTables
(
@SearchStr nvarchar(100)
)
AS
BEGIN

    CREATE TABLE #Results (ColumnName nvarchar(370), ColumnValue nvarchar(3630))

    SET NOCOUNT ON

    DECLARE @TableName nvarchar(256), @ColumnName nvarchar(128), @SearchStr2 nvarchar(110)
    SET  @TableName = ''
    SET @SearchStr2 = QUOTENAME('%' + @SearchStr + '%','''')

    WHILE @TableName IS NOT NULL

    BEGIN
        SET @ColumnName = ''
        SET @TableName = 
        (
            SELECT MIN(QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME))
            FROM     INFORMATION_SCHEMA.TABLES
            WHERE         TABLE_TYPE = 'BASE TABLE'
                AND    QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME) > @TableName
                AND    OBJECTPROPERTY(
                        OBJECT_ID(
                            QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME)
                             ), 'IsMSShipped'
                               ) = 0
        )

        WHILE (@TableName IS NOT NULL) AND (@ColumnName IS NOT NULL)

        BEGIN
            SET @ColumnName =
            (
                SELECT MIN(QUOTENAME(COLUMN_NAME))
                FROM     INFORMATION_SCHEMA.COLUMNS
                WHERE         TABLE_SCHEMA    = PARSENAME(@TableName, 2)
                    AND    TABLE_NAME    = PARSENAME(@TableName, 1)
                    AND    DATA_TYPE IN ('char', 'varchar', 'nchar', 'nvarchar', 'int', 'decimal')
                    AND    QUOTENAME(COLUMN_NAME) > @ColumnName
            )

            IF @ColumnName IS NOT NULL

            BEGIN
                INSERT INTO #Results
                EXEC
                (
                    'SELECT ''' + @TableName + '.' + @ColumnName + ''', LEFT(' + @ColumnName + ', 3630) 
                    FROM ' + @TableName + 'WITH (NOLOCK) ' +
                    ' WHERE ' + @ColumnName + ' LIKE ' + @SearchStr2
                )
            END
        END    
    END

    SELECT ColumnName, ColumnValue FROM #Results
END

21

สายไปหน่อย แต่คุณสามารถค้นหาสตริงด้วยข้อความค้นหานี้ได้อย่างง่ายดาย

DECLARE
@search_string  VARCHAR(100),
@table_name     SYSNAME,
@table_id       INT,
@column_name    SYSNAME,
@sql_string     VARCHAR(2000)

SET @search_string = 'StringtoSearch'

DECLARE tables_cur CURSOR FOR SELECT ss.name +'.'+ so.name [name], object_id FROM sys.objects so INNER JOIN sys.schemas ss ON so.schema_id = ss.schema_id WHERE  type = 'U'

OPEN tables_cur

FETCH NEXT FROM tables_cur INTO @table_name, @table_id

WHILE (@@FETCH_STATUS = 0)
BEGIN
    DECLARE columns_cur CURSOR FOR SELECT name FROM sys.columns WHERE object_id = @table_id 
        AND system_type_id IN (167, 175, 231, 239)

    OPEN columns_cur

    FETCH NEXT FROM columns_cur INTO @column_name
        WHILE (@@FETCH_STATUS = 0)
        BEGIN
            SET @sql_string = 'IF EXISTS (SELECT * FROM ' + @table_name + ' WHERE [' + @column_name + '] 
            LIKE ''%' + @search_string + '%'') PRINT ''' + @table_name + ', ' + @column_name + ''''

            EXECUTE(@sql_string)

        FETCH NEXT FROM columns_cur INTO @column_name
        END

    CLOSE columns_cur

DEALLOCATE columns_cur

FETCH NEXT FROM tables_cur INTO @table_name, @table_id
END

CLOSE tables_cur
DEALLOCATE tables_cur

สคริปต์นี้ล้มเหลวเมื่อสคีมาของตารางไม่ใช่ "dbo" โชคไม่ดีที่ตารางส่วนใหญ่เป็น DB มีสคีมาที่แตกต่างกัน ทำงานได้ดีสำหรับคู่ของ "dbo"
ปล้น

เพิ่งแก้ไขเคอร์เซอร์เริ่มต้นไปที่ INNER JOIN บนตารางสกีมาดังนั้นตอนนี้มันจะมีสคีมาในชื่อตารางเมื่อมันมาใช้
ปล้น

ฉันได้เพิ่ม 99 (ntext) ไปยังรายการ system_types เพื่อค้นหาตามปัจจุบันมันจะละเว้นฟิลด์ xml ทั้งหมด `AND system_type_id IN (167, 175, 231, 239, 99)`
ปล้น

3

ฉันเขียน SP สำหรับสิ่งนี้ซึ่งส่งกลับผลลัพธ์การค้นหาในรูปแบบของชื่อตารางชื่อคอลัมน์ที่พบสตริงคำหลักการค้นหาและการค้นหาแถวที่เกี่ยวข้องดังแสดงในภาพด้านล่าง

ตัวอย่างผลการค้นหา

นี่อาจไม่ใช่โซลูชันที่มีประสิทธิภาพที่สุด แต่คุณสามารถแก้ไขและใช้งานได้ตามความต้องการของคุณ

IF OBJECT_ID('sp_KeywordSearch', 'P') IS NOT NULL
    DROP PROC sp_KeywordSearch
GO

CREATE PROCEDURE sp_KeywordSearch @KeyWord NVARCHAR(100)
AS
BEGIN
    DECLARE @Result TABLE
        (TableName NVARCHAR(300),
         ColumnName NVARCHAR(MAX))

    DECLARE @Sql NVARCHAR(MAX),
        @TableName NVARCHAR(300),
        @ColumnName NVARCHAR(300),
        @Count INT

    DECLARE @tableCursor CURSOR

    SET @tableCursor = CURSOR LOCAL SCROLL FOR
    SELECT  N'SELECT @Count = COUNT(1) FROM [dbo].[' + T.TABLE_NAME + '] WITH (NOLOCK) WHERE CAST([' + C.COLUMN_NAME +
            '] AS NVARCHAR(MAX)) LIKE ''%' + @KeyWord + N'%''',
            T.TABLE_NAME,
            C.COLUMN_NAME
    FROM    INFORMATION_SCHEMA.TABLES AS T WITH (NOLOCK)
    INNER JOIN INFORMATION_SCHEMA.COLUMNS AS C WITH (NOLOCK)
    ON      T.TABLE_SCHEMA = C.TABLE_SCHEMA AND
            T.TABLE_NAME = C.TABLE_NAME
    WHERE   T.TABLE_TYPE = 'BASE TABLE' AND
            C.TABLE_SCHEMA = 'dbo' AND
            C.DATA_TYPE NOT IN ('image', 'timestamp')

    OPEN @tableCursor
    FETCH NEXT FROM @tableCursor INTO @Sql, @TableName, @ColumnName

    WHILE (@@FETCH_STATUS = 0)
    BEGIN
        SET @Count = 0

        EXEC sys.sp_executesql
            @Sql,
            N'@Count INT OUTPUT',
            @Count OUTPUT

        IF @Count > 0
        BEGIN
            INSERT  INTO @Result
                    (TableName, ColumnName)
            VALUES  (@TableName, @ColumnName)
        END

        FETCH NEXT FROM @tableCursor INTO @Sql, @TableName, @ColumnName
    END

    CLOSE @tableCursor
    DEALLOCATE @tableCursor

    SET @tableCursor = CURSOR LOCAL SCROLL FOR
    SELECT  SUBSTRING(TB.Sql, 1, LEN(TB.Sql) - 3) AS Sql, TB.TableName, SUBSTRING(TB.Columns, 1, LEN(TB.Columns) - 1) AS Columns
    FROM    (SELECT R.TableName, (SELECT R2.ColumnName + ', ' FROM @Result AS R2 WHERE R.TableName = R2.TableName FOR XML PATH('')) AS Columns,
                    'SELECT * FROM ' + R.TableName + ' WITH (NOLOCK) WHERE ' +
                    (SELECT 'CAST(' + R2.ColumnName + ' AS NVARCHAR(MAX)) LIKE ''%' + @KeyWord + '%'' OR '
                     FROM   @Result AS R2
                     WHERE  R.TableName = R2.TableName
                    FOR
                     XML PATH('')) AS Sql
             FROM   @Result AS R
             GROUP BY R.TableName) TB
    ORDER BY TB.Sql

    OPEN @tableCursor
    FETCH NEXT FROM @tableCursor INTO @Sql, @TableName, @ColumnName

    WHILE (@@FETCH_STATUS = 0)
    BEGIN
        PRINT @Sql
        SELECT  @TableName AS [Table],
                @ColumnName AS Columns
        EXEC(@Sql)

        FETCH NEXT FROM @tableCursor INTO @Sql, @TableName, @ColumnName
    END

    CLOSE @tableCursor
    DEALLOCATE @tableCursor

END

ตำแหน่งที่จะกำหนดสตริงการค้นหาของเรา
Dupinder Singh

1

ไม่จำเป็นต้องใช้การวนซ้ำซ้อนกัน (การวนลูปนอกผ่านตารางและการวนซ้ำภายในผ่านคอลัมน์ตารางทั้งหมด) หนึ่งสามารถดึงข้อมูลทั้งหมด (หรือเลือกโดยพลการ / กรอง) คอลัมน์ชุดค่าผสมจากINFORMATION_SCHEMA.COLUMNSและในหนึ่งวงเพียงแค่ผ่าน (ค้นหา) พวกเขาทั้งหมด:

DECLARE @search VARCHAR(100), @table SYSNAME, @column SYSNAME

DECLARE curTabCol CURSOR FOR
    SELECT c.TABLE_SCHEMA + '.' + c.TABLE_NAME, c.COLUMN_NAME
    FROM INFORMATION_SCHEMA.COLUMNS c
    JOIN INFORMATION_SCHEMA.TABLES t 
      ON t.TABLE_NAME=c.TABLE_NAME AND t.TABLE_TYPE='BASE TABLE' -- avoid views
    WHERE c.DATA_TYPE IN ('varchar','nvarchar') -- searching only in these column types
    --AND c.COLUMN_NAME IN ('NAME','DESCRIPTION') -- searching only in these column names

SET @search='john'

OPEN curTabCol
FETCH NEXT FROM curTabCol INTO @table, @column

WHILE (@@FETCH_STATUS = 0)
BEGIN
    EXECUTE('IF EXISTS 
             (SELECT * FROM ' + @table + ' WHERE ' + @column + ' = ''' + @search + ''') 
             PRINT ''' + @table + '.' + @column + '''')
    FETCH NEXT FROM curTabCol INTO @table, @column
END

CLOSE curTabCol
DEALLOCATE curTabCol

ฉันไม่รู้ว่าคุณหมายถึงอะไร แต่คำถามนี้ใช้ได้ดี เคอร์เซอร์curTabColเลือก table.column ทั้งหมดที่น่าสนใจ จากนั้นค้นหาคอลัมน์ทั้งหมดเหล่านี้เพื่อหาสตริงที่กำหนดในตัวแปร @search หากสตริงดังกล่าว (แน่นอน) ในแบบสอบถาม SQL นี้ (มิฉะนั้นใช้ LIKE '% ... %') อยู่ชุดชื่อคอลัมน์และคอลัมน์จะถูกพิมพ์ออกมา
sbrbot

0

สิ่งนี้มีประโยชน์มาก ฉันต้องการนำเข้าฟังก์ชั่นนี้ไปยังฐานข้อมูล Postgre SQL คิดว่าฉันจะแบ่งปันกับทุกคนที่มีความสนใจ จะมีพวกเขาไม่กี่ชั่วโมง หมายเหตุ: ฟังก์ชันนี้สร้างรายการคำสั่ง SQL ที่สามารถคัดลอกและดำเนินการในฐานข้อมูล Postgre บางทีใครบางคนที่ฉลาดกว่านั้นฉันสามารถให้ Postgre สร้างและดำเนินการคำสั่งทั้งหมดในที่เดียว

CREATE OR REPLACE FUNCTION SearchAllTables(_search text) RETURNS TABLE( txt text ) as $funct$
    DECLARE __COUNT int;
    __SQL text;
BEGIN
    EXECUTE 'SELECT COUNT(0) FROM INFORMATION_SCHEMA.COLUMNS
                    WHERE    DATA_TYPE = ''text'' 
                    AND          table_schema = ''public'' ' INTO __COUNT;

    RETURN QUERY 
        SELECT CASE WHEN ROW_NUMBER() OVER (ORDER BY table_name) < __COUNT THEN 
            'SELECT ''' || table_name ||'.'|| column_name || ''' AS tbl, "'  || column_name || '" AS col FROM "public"."' || "table_name" || '" WHERE "'|| "column_name" || '" ILIKE ''%' || _search  || '%'' UNION ALL' 
            ELSE 
            'SELECT ''' || table_name ||'.'|| column_name || ''' AS tbl, "'  || column_name || '" AS col FROM "public"."' || "table_name" || '" WHERE "'|| "column_name" || '" ILIKE ''%' || _search  || '%'''
        END AS txt

                    FROM     INFORMATION_SCHEMA.COLUMNS
                    WHERE    DATA_TYPE = 'text' 
                    AND          table_schema = 'public';
END
$funct$ LANGUAGE plpgsql;

0

คำตอบที่ถูกกล่าวถึงในโพสต์นี้แล้วหลายครั้งฉันได้นำเล็กน้อยเพราะฉันต้องการค้นหาในตารางเดียวเท่านั้น:

(และทำให้อินพุตสำหรับชื่อตารางง่ายขึ้นอีกเล็กน้อย)

ALTER PROC dbo.db_compare_SearchAllTables_sp
(
    @SearchStr nvarchar(100),
    @TableName nvarchar(256) = ''
)
AS
BEGIN
        if PARSENAME(@TableName, 2) is null
            set @TableName = 'dbo.' + QUOTENAME(@TableName, '"')

    declare @results TABLE(ColumnName nvarchar(370), ColumnValue nvarchar(3630))

    SET NOCOUNT ON

    DECLARE @ColumnName nvarchar(128) = '', @SearchStr2 nvarchar(110)
    SET @SearchStr2 = QUOTENAME('%' + @SearchStr + '%','''')
        IF @TableName <> ''
        WHILE (@TableName IS NOT NULL) AND (@ColumnName IS NOT NULL)
        BEGIN
            SET @ColumnName =
            (
                SELECT MIN(QUOTENAME(COLUMN_NAME))
                FROM     INFORMATION_SCHEMA.COLUMNS
                WHERE         TABLE_SCHEMA    = PARSENAME(@TableName, 2)
                    AND    TABLE_NAME    = PARSENAME(@TableName, 1)
                    AND    DATA_TYPE IN ('char', 'varchar', 'nchar', 'nvarchar')
                    AND    QUOTENAME(COLUMN_NAME) > @ColumnName
            )

            IF @ColumnName IS NOT NULL
            BEGIN
                INSERT INTO @results
                EXEC
                (
                    'SELECT ''' + @TableName + '.' + @ColumnName + ''', LEFT(' + @ColumnName + ', 3630) 
                    FROM ' + @TableName + ' WITH (NOLOCK) ' +
                    ' WHERE ' + @ColumnName + ' LIKE ' + @SearchStr2
                )
            END
        END    
        ELSE
    WHILE @TableName IS NOT NULL
    BEGIN
        SET @ColumnName = ''
        SET @TableName = 
        (
            SELECT MIN(QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME))
            FROM     INFORMATION_SCHEMA.TABLES
            WHERE         TABLE_TYPE = 'BASE TABLE'
                AND    QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME) > @TableName
                AND    OBJECTPROPERTY(
                        OBJECT_ID(
                            QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME)
                             ), 'IsMSShipped'
                               ) = 0
        )

        WHILE (@TableName IS NOT NULL) AND (@ColumnName IS NOT NULL)
        BEGIN
            SET @ColumnName =
            (
                SELECT MIN(QUOTENAME(COLUMN_NAME))
                FROM     INFORMATION_SCHEMA.COLUMNS
                WHERE         TABLE_SCHEMA    = PARSENAME(@TableName, 2)
                    AND    TABLE_NAME    = PARSENAME(@TableName, 1)
                    AND    DATA_TYPE IN ('char', 'varchar', 'nchar', 'nvarchar')
                    AND    QUOTENAME(COLUMN_NAME) > @ColumnName
            )

            IF @ColumnName IS NOT NULL
            BEGIN
                INSERT INTO @results
                EXEC
                (
                    'SELECT ''' + @TableName + '.' + @ColumnName + ''', LEFT(' + @ColumnName + ', 3630) 
                    FROM ' + @TableName + ' WITH (NOLOCK) ' +
                    ' WHERE ' + @ColumnName + ' LIKE ' + @SearchStr2
                )
            END
        END    
    END

    SELECT ColumnName, ColumnValue FROM @results
END
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.