Suchen Sie einen Wert an einer beliebigen Stelle in einer Datenbank

253

Wie finde ich mit einem # heraus, in welcher Tabelle und Spalte es sich befindet?

Es ist mir egal, ob es schnell ist, es muss nur funktionieren.

Allain Lalonde
quelle
1
Sie möchten alle Spalten / Zeilen nach einer bestimmten Nummer durchsuchen? Können Sie es auf numerische Spalten beschränken? Ganzzahlige Spalten? Identitätsspalten?
Michael Haren
1
Alle Spalten wären am besten, aber numerisch wäre ausreichend. Einzugsspalten sind zu spezifisch
Allain Lalonde
1
Sie müssen wahrscheinlich ein kurzes Skript schreiben, um die Metadaten (in diesem Fall eine Liste von Tabellen / Spalten) aus der Datenbank abzufragen und eine Reihe von Select-Anweisungen auszugeben, die nach dem Wert suchen.
Draemon
9
Das ist uralt, aber warum nicht einfach eine Müllkippe machen und die Müllkippe greifen?
Baordog
3
phpmyadmin erlauben dies sehr einfach
Matoeil

Antworten:

284

Dies könnte Ihnen helfen . - von Narayana Vyas. Es durchsucht alle Spalten aller Tabellen in einer bestimmten Datenbank. Ich habe es schon einmal benutzt und es funktioniert.

Dies ist der gespeicherte Prozess über den obigen Link. Die einzige Änderung, die ich vorgenommen habe, war das Ersetzen einer Tabellenvariablen durch die temporäre Tabelle, damit Sie nicht jedes Mal daran denken müssen, sie zu löschen.

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

-- Copyright © 2002 Narayana Vyas Kondreddi. All rights reserved.
-- Purpose: To search all columns of all tables for a given search string
-- Written by: Narayana Vyas Kondreddi
-- Site: http://vyaskn.tripod.com
-- Tested on: SQL Server 7.0 and SQL Server 2000
-- Date modified: 28th July 2002 22:50 GMT

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')
                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
END
Ta01
quelle
17
Zu Ihrer Information, dieses Skript durchsucht nur Textfelder, keine Zahlenfelder. In meinem Fall hat es funktioniert, weil die Entwickler eine Nummer in einem Varchar gespeichert haben, aber im Allgemeinen funktioniert es nicht, Nummern zu finden.
Allain Lalonde
1
Können wir dies mit einer einzelnen Abfrage anstelle einer gespeicherten Prozedur tun?
Freakyuser
6
Wie kann jemand ein einfaches SQL-Verfahren urheberrechtlich schützen? es verblüfft mich
David Andrei Ned
3
@ DavidAndreiNed, er kann nicht, zumindest nicht so wie er. Ich bin mir nicht sicher, ob dies 2009 der Fall war, aber heutzutage lauten die Antworten CC-BY-SA 3.0
Arturo Torres Sánchez
2
Wo im Skript soll ich die Zeichenfolge platzieren, die ich finden möchte? im Set @ SearchStr2?
Randy Quackers
77

Wenn Sie eine solche Suche nur einmal ausführen müssen, können Sie wahrscheinlich eines der Skripte verwenden, die bereits in anderen Antworten angezeigt werden. Ansonsten würde ich empfehlen, dafür ApexSQL Search zu verwenden. Es ist ein kostenloses SSMS-Add-In und hat mir sehr viel Zeit gespart.

Bevor Sie eines der Skripte ausführen, sollten Sie es basierend auf dem Datentyp anpassen, den Sie suchen möchten. Wenn Sie wissen, dass Sie nach einer Datums- / Uhrzeitspalte suchen, müssen Sie die nvarchar-Spalten nicht durchsuchen. Dies beschleunigt alle oben genannten Abfragen.

David Smithers
quelle
1
Ich habe eine Weile gebraucht, um herauszufinden, wo ich das tun soll. Jeder, der diesen Kommentar liest, kann dies in der Abfrage tun, indem er dieses Stück von der akzeptierten Antwort ändert: AND DATA_TYPE IN ('char', 'varchar', 'nchar', 'nvarchar ')
Taylor Brown
ApexSQL Search ist ein hervorragendes Tool. Ich denke, dies sollte die beste Antwort sein, da ein Tool ein abstraktes Konzept ist, bei dem keine schmutzigen Skripte verwendet werden.
nützlichBee
Beachten Sie auch, dass ApexSQL Sie zwar für einige ihrer Tools bezahlen lässt, dieses jedoch kostenlos ist. Sie müssen ihnen nur Ihre E-Mail geben. Ein großartiges Tool, das mir sehr helfen wird, wenn ich in undokumentierten Datenbanken von
Drittanbietern grabe
Dies sollte meiner Meinung nach die beste Antwort sein. Die Möglichkeit, nach verschiedenen Datentypen zu filtern, ist großartig. Der einzige Nachteil , den ich habe darüber ist, dass die GUI für die Auswahl / Abwahl „ welche Tabellen in suchen“ ist eine Checkbox-Liste mit NO Aktivieren / Deaktivieren-ALL - Option oder Multi-Select-and-Toggle - Fähigkeit. Wenn Sie beispielsweise alle Systemobjekte deaktivieren möchten (die standardmäßig * aktiviert * sind, was albern ist), müssen Sie eine Übung mit Leerzeichen-Pfeil nach unten-Leerzeichen-Pfeil nach unten * ad-nauseum * durchführen . Aber zum Glück ist es effizient genug, dass Sie nicht müssen. Lass es einfach suchen und mach sein Ding!
NateJ
ApexSQL Search ist definitiv die beste Lösung. Ich habe gerade versucht, mithilfe des Skripts eine E-Mail-Adresse in einer Datenbank zu finden - 8:30 Minuten später gab ich auf. Nach der Installation von ApexSQL Search habe ich nach genau der gleichen Zeichenfolge gesucht und sie 31 Mal in 11 Tabellen gefunden. Ich habe es nicht genau
gemessen
74

Basierend auf der Antwort von bnkdev habe ich den Code von Narayana so geändert , dass alle Spalten durchsucht werden, auch numerische.

Es läuft langsamer, aber diese Version findet tatsächlich alle Übereinstimmungen, nicht nur die in Textspalten.

Ich kann diesem Kerl nicht genug danken. Ersparte mir Tage der Suche von Hand!

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

-- Copyright © 2002 Narayana Vyas Kondreddi. All rights reserved.
-- Purpose: To search all columns of all tables for a given search string
-- Written by: Narayana Vyas Kondreddi
-- Site: http://vyaskn.tripod.com
-- Tested on: SQL Server 7.0 and SQL Server 2000
-- Date modified: 28th July 2002 22:50 GMT


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 QUOTENAME(COLUMN_NAME) > @ColumnName
        )

        IF @ColumnName IS NOT NULL
        BEGIN
            INSERT INTO #Results
            EXEC
            (
                'SELECT ''' + @TableName + '.' + @ColumnName + ''', LEFT(CONVERT(varchar(max), ' + @ColumnName + '), 3630) 
                FROM ' + @TableName + ' (NOLOCK) ' +
                ' WHERE CONVERT(varchar(max), ' + @ColumnName + ') LIKE ' + @SearchStr2
            )
        END
    END 
END

SELECT ColumnName, ColumnValue FROM #Results
END
Allain Lalonde
quelle
5
bearbeitet, um den Fehler "Unzureichender Ergebnisbereich zum Konvertieren des eindeutigen Identifizierungswerts in char" zu beheben. Dies funktioniert jetzt auch für XML-Spalten.
Chris
1
Hier ist eine neue: Eine explizite Konvertierung vom Datentypbild in varchar (max) ist nicht zulässig. Ich werde versuchen, dies selbst zu beheben, aber wenn mich jemand schlägt, lass es mich wissen, danke!
Taylor Brown
5
Ok, ich habe gerade ein Stück aus dem Code von kd7 hinzugefügt, um nur nach den Datentypen zu suchen, nach denen ich suche, wodurch diese Bildspalten aus meiner Suche herausfallen und der Fehler verschwindet ... UND DATA_TYPE NOT IN ('image')
Taylor Brown
1
Erhalten "Konvertierung eines oder mehrerer Zeichen von XML in Zielsortierung unmöglich Fehler" in SQL Server 2014.
Chetan Mehra
Sie könnten einfach verschiedene Versionen für verschiedene Variablentypen haben. Auf diese Weise wirfst du nicht und es läuft schneller. Sie können die Dateitypen auch mit der Suche nach gießbaren Typen vergleichen. Eine Ganzzahl kann sich in einem Varchar-Feld befinden.
SQLMason
38

Dies ist meine unabhängige Sicht auf diese Frage, die ich für meine eigene Arbeit verwende. Es funktioniert in SQL2000 und höher, ermöglicht Platzhalter und Spaltenfilterung und durchsucht die meisten normalen Datentypen.

Eine Pseudocode-Beschreibung könnte sein select * from * where any like 'foo'

--------------------------------------------------------------------------------
-- Search all columns in all tables in a database for a string.
-- Does not search: image, sql_variant or user-defined types.
-- Exact search always for money and smallmoney; no wildcards for matching these.
--------------------------------------------------------------------------------
declare @SearchTerm nvarchar(4000) -- Can be max for SQL2005+
declare @ColumnName sysname

--------------------------------------------------------------------------------
-- SET THESE!
--------------------------------------------------------------------------------
set @SearchTerm = N'foo' -- Term to be searched for, wildcards okay
set @ColumnName = N'' -- Use to restrict the search to certain columns, wildcards okay, null or empty string for all cols
--------------------------------------------------------------------------------
-- END SET
--------------------------------------------------------------------------------

set nocount on

declare @TabCols table (
      id int not null primary key identity
    , table_schema sysname not null
    , table_name sysname not null
    , column_name sysname not null
    , data_type sysname not null
)
insert into @TabCols (table_schema, table_name, column_name, data_type)
    select t.TABLE_SCHEMA, c.TABLE_NAME, c.COLUMN_NAME, c.DATA_TYPE
    from INFORMATION_SCHEMA.TABLES t
        join INFORMATION_SCHEMA.COLUMNS c on t.TABLE_SCHEMA = c.TABLE_SCHEMA
            and t.TABLE_NAME = c.TABLE_NAME
    where 1 = 1
        and t.TABLE_TYPE = 'base table'
        and c.DATA_TYPE not in ('image', 'sql_variant')
        and c.COLUMN_NAME like case when len(@ColumnName) > 0 then @ColumnName else '%' end
    order by c.TABLE_NAME, c.ORDINAL_POSITION

declare
      @table_schema sysname
    , @table_name sysname
    , @column_name sysname
    , @data_type sysname
    , @exists nvarchar(4000) -- Can be max for SQL2005+
    , @sql nvarchar(4000) -- Can be max for SQL2005+
    , @where nvarchar(4000) -- Can be max for SQL2005+
    , @run nvarchar(4000) -- Can be max for SQL2005+

while exists (select null from @TabCols) begin

    select top 1
          @table_schema = table_schema
        , @table_name = table_name
        , @exists = 'select null from [' + table_schema + '].[' + table_name + '] where 1 = 0'
        , @sql = 'select ''' + '[' + table_schema + '].[' + table_name + ']' + ''' as TABLE_NAME, * from [' + table_schema + '].[' + table_name + '] where 1 = 0'
        , @where = ''
    from @TabCols
    order by id

    while exists (select null from @TabCols where table_schema = @table_schema and table_name = @table_name) begin

        select top 1
              @column_name = column_name
            , @data_type = data_type
        from @TabCols
        where table_schema = @table_schema
            and table_name = @table_name
        order by id

        -- Special case for money
        if @data_type in ('money', 'smallmoney') begin
            if isnumeric(@SearchTerm) = 1 begin
                set @where = @where + ' or [' + @column_name + '] = cast(''' + @SearchTerm + ''' as ' + @data_type + ')' -- could also cast the column as varchar for wildcards
            end
        end
        -- Special case for xml
        else if @data_type = 'xml' begin
            set @where = @where + ' or cast([' + @column_name + '] as nvarchar(max)) like ''' + @SearchTerm + ''''
        end
        -- Special case for date
        else if @data_type in ('date', 'datetime', 'datetime2', 'datetimeoffset', 'smalldatetime', 'time') begin
            set @where = @where + ' or convert(nvarchar(50), [' + @column_name + '], 121) like ''' + @SearchTerm + ''''
        end
        -- Search all other types
        else begin
            set @where = @where + ' or [' + @column_name + '] like ''' + @SearchTerm + ''''
        end

        delete from @TabCols where table_schema = @table_schema and table_name = @table_name and column_name = @column_name

    end

    set @run = 'if exists(' + @exists + @where + ') begin ' + @sql + @where + ' print ''' + @table_name + ''' end'
    print @run
    exec sp_executesql @run

end

set nocount off

Ich stelle es nicht in Proc-Form, da ich es nicht über Hunderte von DBs hinweg verwalten möchte und es sowieso wirklich für Ad-hoc-Arbeiten ist. Bitte zögern Sie nicht, Fehlerbehebungen zu kommentieren.

Tim Lehner
quelle
Danke, aber ich bekomme nichts als Syntaxfehler in phpMyAdmin. Hat sich mit SQL etwas geändert, seit dies geschrieben wurde?
NoBugs
3
@NoBugs Dies ist in T-SQL für SQL Server geschrieben.
Tim Lehner
@NoBugs: Sie müssen den Code in Ihre eigene gespeicherte Prozedur oder eine andere Funktion kapseln.
Fandango68
2
Um die Ausgabe zu verdeutlichen, wird die Registerkarte Ergebnisse nur geöffnet, wenn der Suchbegriff gefunden wird, wenn dies in Microsoft SQL Server Management Studio ausgeführt wird. Wird der Suchbegriff nicht gefunden, wird nur die Registerkarte Nachrichten mit den ausgeführten Suchanweisungen geöffnet. Die Registerkarte "Nachricht" enthält keine Ergebnisse, wird jedoch zusammen mit der Registerkarte "Ergebnisse" geöffnet, wenn der Suchbegriff gefunden wird.
Gakera
1
Das ist toll! Arbeiten mit einer gekauften App, bei der ich keine benutzerdefinierten Datenbankobjekte in ihrem Schema speichern möchte. Vielen Dank!
Jeff
20

Ich habe die Antwort von Allain Lalonde optimiert ( https://stackoverflow.com/a/436676/412368 ). Numerische Werte werden weiterhin unterstützt. Sollte ungefähr 4-5 mal schneller sein (1:03 vs 4:30), getestet auf einem Desktop mit einer 7 GB Datenbank. http://developer.azurewebsites.net/2015/01/mssql-searchalltables/

IF OBJECT_ID ('dbo.SearchAllTables', 'P') IS NOT NULL 
    DROP PROCEDURE dbo.SearchAllTables;
GO

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

-- Copyright © 2002 Narayana Vyas Kondreddi. All rights reserved.
-- Purpose: To search all columns of all tables for a given search string
-- Written by: Narayana Vyas Kondreddi
-- Site: http://vyaskn.tripod.com
-- Customized and modified: 2014-01-21
-- Tested on: SQL Server 2008 R2

DECLARE @Results TABLE(ColumnName nvarchar(370), ColumnValue nvarchar(3630))

SET NOCOUNT ON

DECLARE @TableName nvarchar(256)
DECLARE @ColumnName nvarchar(128)
DECLARE @DataType nvarchar(128)

DECLARE @SearchStr2 nvarchar(110)
DECLARE @SearchDecimal decimal(38,19)
DECLARE @Query nvarchar(4000)
SET @SearchStr2 = QUOTENAME('%' + @SearchStr + '%', '''')
SET @SearchDecimal = CASE WHEN ISNUMERIC(@SearchStr) = 1 THEN CONVERT(decimal(38,19), @SearchStr) ELSE NULL END
PRINT '@SearchStr2: ' + @SearchStr2
PRINT '@SearchDecimal: ' + CAST(@SearchDecimal AS nvarchar)

SET @TableName = ''
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))
                    DATA_TYPE
            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', 'bigint', 'tinyint', 'numeric', 'decimal')
                AND QUOTENAME(COLUMN_NAME) > @ColumnName
        )
        SET @DataType =
        (
            SELECT DATA_TYPE
            FROM    INFORMATION_SCHEMA.COLUMNS
            WHERE       TABLE_SCHEMA    = PARSENAME(@TableName, 2)
                AND TABLE_NAME  = PARSENAME(@TableName, 1)
                AND QUOTENAME(COLUMN_NAME) = @ColumnName
        )
        PRINT @TableName + '.' + @ColumnName + ' (' + @DataType + ')'

        IF @ColumnName IS NOT NULL
        BEGIN
            IF @DataType IN ('int', 'bigint', 'tinyint', 'numeric', 'decimal')
            BEGIN
                IF @SearchDecimal IS NOT NULL
                BEGIN
                    SET @Query = 'SELECT ''' + @TableName + '.' + @ColumnName + ''', LEFT(CAST(' + @ColumnName + ' AS nvarchar(110)), 3630) ' +
                                 'FROM ' + @TableName + ' (NOLOCK) ' +
                                 ' WHERE ' + @ColumnName + ' = ' + CAST(@SearchDecimal AS nvarchar)
                    PRINT '    ' + @Query
                    INSERT INTO @Results
                    EXEC (@Query)
                END
            END
            ELSE
            BEGIN
                SET @Query = 'SELECT ''' + @TableName + '.' + @ColumnName + ''', LEFT(' + @ColumnName + ', 3630) ' +
                             'FROM ' + @TableName + ' (NOLOCK) ' +
                             ' WHERE ' + @ColumnName + ' LIKE ' + @SearchStr2
                PRINT '    ' + @Query
                INSERT INTO @Results
                EXEC (@Query)
            END
        END
    END 
END

SELECT ColumnName, ColumnValue FROM @Results
END
Lauri Lubi
quelle
5

Auf diese Weise kann ich diese Frage lösen. Getestet auf SQLServer2008R2

CREATE PROC SearchAllTables
@SearchStr nvarchar(100)
AS
BEGIN
DECLARE @dml nvarchar(max) = N''        
IF OBJECT_ID('tempdb.dbo.#Results') IS NOT NULL DROP TABLE dbo.#Results
CREATE TABLE dbo.#Results
 ([tablename] nvarchar(100), 
  [ColumnName] nvarchar(100), 
  [Value] nvarchar(max))  
SELECT @dml += ' SELECT ''' + s.name + '.' + t.name + ''' AS [tablename], ''' + 
                c.name + ''' AS [ColumnName], CAST(' + QUOTENAME(c.name) + 
               ' AS nvarchar(max)) AS [Value] FROM ' + QUOTENAME(s.name) + '.' + QUOTENAME(t.name) +
               ' (NOLOCK) WHERE CAST(' + QUOTENAME(c.name) + ' AS nvarchar(max)) LIKE ' + '''%' + @SearchStr + '%'''
FROM sys.schemas s JOIN sys.tables t ON s.schema_id = t.schema_id
                   JOIN sys.columns c ON t.object_id = c.object_id
                   JOIN sys.types ty ON c.system_type_id = ty.system_type_id AND c .user_type_id = ty .user_type_id
WHERE t.is_ms_shipped = 0 AND ty.name NOT IN ('timestamp', 'image', 'sql_variant')

INSERT dbo.#Results
EXEC sp_executesql @dml

SELECT *
FROM dbo.#Results
END
Aleksandr Fedorenko
quelle
4

Danke für das wirklich nützliche Skript.

Möglicherweise müssen Sie dem Code die folgende Änderung hinzufügen, wenn Ihre Tabellen nicht konvertierbare Felder enthalten:

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 NOT IN ('text', 'image', 'ntext')                 
            AND QUOTENAME(COLUMN_NAME) > @ColumnName
    )

Chris

Rab_
quelle
3

Ich habe eine Lösung von vor einiger Zeit, die ich weiter verbessert habe. Sucht auch in XML-Spalten, wenn Sie dazu aufgefordert werden, oder sucht nach ganzzahligen Werten, wenn nur eine Ganzzahlzeichenfolge angegeben wird.

/* Reto Egeter, fullparam.wordpress.com */

DECLARE @SearchStrTableName nvarchar(255), @SearchStrColumnName nvarchar(255), @SearchStrColumnValue nvarchar(255), @SearchStrInXML bit, @FullRowResult bit, @FullRowResultRows int
SET @SearchStrColumnValue = '%searchthis%' /* use LIKE syntax */
SET @FullRowResult = 1
SET @FullRowResultRows = 3
SET @SearchStrTableName = NULL /* NULL for all tables, uses LIKE syntax */
SET @SearchStrColumnName = NULL /* NULL for all columns, uses LIKE syntax */
SET @SearchStrInXML = 0 /* Searching XML data may be slow */

IF OBJECT_ID('tempdb..#Results') IS NOT NULL DROP TABLE #Results
CREATE TABLE #Results (TableName nvarchar(128), ColumnName nvarchar(128), ColumnValue nvarchar(max),ColumnType nvarchar(20))

SET NOCOUNT ON

DECLARE @TableName nvarchar(256) = '',@ColumnName nvarchar(128),@ColumnType nvarchar(20), @QuotedSearchStrColumnValue nvarchar(110), @QuotedSearchStrColumnName nvarchar(110)
SET @QuotedSearchStrColumnValue = QUOTENAME(@SearchStrColumnValue,'''')
DECLARE @ColumnNameTable TABLE (COLUMN_NAME nvarchar(128),DATA_TYPE nvarchar(20))

WHILE @TableName IS NOT NULL
BEGIN
SET @TableName =
(
SELECT MIN(QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME))
FROM INFORMATION_SCHEMA.TABLES
WHERE TABLE_TYPE = 'BASE TABLE'
AND TABLE_NAME LIKE COALESCE(@SearchStrTableName,TABLE_NAME)
AND QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME) > @TableName
AND OBJECTPROPERTY(OBJECT_ID(QUOTENAME(TABLE_SCHEMA) + '.' + QUOTENAME(TABLE_NAME)), 'IsMSShipped') = 0
)
IF @TableName IS NOT NULL
BEGIN
DECLARE @sql VARCHAR(MAX)
SET @sql = 'SELECT QUOTENAME(COLUMN_NAME),DATA_TYPE
FROM INFORMATION_SCHEMA.COLUMNS
WHERE TABLE_SCHEMA = PARSENAME(''' + @TableName + ''', 2)
AND TABLE_NAME = PARSENAME(''' + @TableName + ''', 1)
AND DATA_TYPE IN (' + CASE WHEN ISNUMERIC(REPLACE(REPLACE(REPLACE(REPLACE(REPLACE(@SearchStrColumnValue,'%',''),'_',''),'[',''),']',''),'-','')) = 1 THEN '''tinyint'',''int'',''smallint'',''bigint'',''numeric'',''decimal'',''smallmoney'',''money'',' ELSE '' END + '''char'',''varchar'',''nchar'',''nvarchar'',''timestamp'',''uniqueidentifier''' + CASE @SearchStrInXML WHEN 1 THEN ',''xml''' ELSE '' END + ')
AND COLUMN_NAME LIKE COALESCE(' + CASE WHEN @SearchStrColumnName IS NULL THEN 'NULL' ELSE '''' + @SearchStrColumnName + '''' END + ',COLUMN_NAME)'
INSERT INTO @ColumnNameTable
EXEC (@sql)
WHILE EXISTS (SELECT TOP 1 COLUMN_NAME FROM @ColumnNameTable)
BEGIN
PRINT @ColumnName
SELECT TOP 1 @ColumnName = COLUMN_NAME,@ColumnType = DATA_TYPE FROM @ColumnNameTable
SET @sql = 'SELECT ''' + @TableName + ''',''' + @ColumnName + ''',' + CASE @ColumnType WHEN 'xml' THEN 'LEFT(CAST(' + @ColumnName + ' AS nvarchar(MAX)), 4096),'''
WHEN 'timestamp' THEN 'master.dbo.fn_varbintohexstr('+ @ColumnName + '),'''
ELSE 'LEFT(' + @ColumnName + ', 4096),''' END + @ColumnType + '''
FROM ' + @TableName + ' (NOLOCK) ' +
' WHERE ' + CASE @ColumnType WHEN 'xml' THEN 'CAST(' + @ColumnName + ' AS nvarchar(MAX))'
WHEN 'timestamp' THEN 'master.dbo.fn_varbintohexstr('+ @ColumnName + ')'
ELSE @ColumnName END + ' LIKE ' + @QuotedSearchStrColumnValue
INSERT INTO #Results
EXEC(@sql)
IF @@ROWCOUNT > 0 IF @FullRowResult = 1
BEGIN
SET @sql = 'SELECT TOP ' + CAST(@FullRowResultRows AS VARCHAR(3)) + ' ''' + @TableName + ''' AS [TableFound],''' + @ColumnName + ''' AS [ColumnFound],''FullRow>'' AS [FullRow>],*' +
' FROM ' + @TableName + ' (NOLOCK) ' +
' WHERE ' + CASE @ColumnType WHEN 'xml' THEN 'CAST(' + @ColumnName + ' AS nvarchar(MAX))'
WHEN 'timestamp' THEN 'master.dbo.fn_varbintohexstr('+ @ColumnName + ')'
ELSE @ColumnName END + ' LIKE ' + @QuotedSearchStrColumnValue
EXEC(@sql)
END
DELETE FROM @ColumnNameTable WHERE COLUMN_NAME = @ColumnName
END 
END
END
SET NOCOUNT OFF

SELECT TableName, ColumnName, ColumnValue, ColumnType, COUNT(*) AS Count FROM #Results
GROUP BY TableName, ColumnName, ColumnValue, ColumnType

Quelle: http://fullparam.wordpress.com/2012/09/07/fck-it-i-am-going-to-search-all-tables-all-collumns/

Regeter
quelle
Es war die einzige Antwort, die mit meinen Privilegien funktionierte, nicht nur nach Zeichenfolgen suchte und nicht mit meinen Tabellen gebrochen wurde.
Pedro Lacerda
3

Ich habe mir einmal ein Werkzeug geschrieben, um genau das zu tun:

a7 SqlTools

Es ist kostenlos und Open Source:

Github Link

Alek Kowalczyk
quelle
2

Hier sehr süße und kleine Lösung:

1) create a store procedure:

create procedure get_table
@find_str varchar(50)
as 
begin
  declare @col_name varchar(500), @tab_name varchar(500);
  declare @find_tab TABLE(table_name varchar(100), column_name varchar(100));

  DECLARE tab_col cursor for 
  select C.name as 'col_name', T.name as tab_name
  from sys.tables as T
  left outer join sys.columns as C on  C.object_id=T.object_id
  left outer join sys.types as TP on  C.system_type_id=TP.system_type_id
  where type='U' 
  and TP.name in('text','ntext','varchar','char','nvarchar','nchar');

  open tab_col
  fetch next from tab_col into @col_name, @tab_name

  while @@FETCH_STATUS = 0
  begin        
    insert into @find_tab 
    exec('select ''' +  @tab_name + ''',''' + @col_name + ''' from ' + @tab_name + 
    ' where ' + @col_name + '=''' + @find_str + ''' group by ' + 
    @col_name + ' having count(*)>0');

    fetch next from tab_col into @col_name, @tab_name;
  end
  CLOSE tab_col;  
  DEALLOCATE tab_col; 
  select table_name, column_name from @find_tab;

end

==========================

2) call procedure by calling store procedure:
exec get_table 'serach_string';
jainvikram444
quelle
1

Ein anderer Weg mit JOIN und CURSOR:

USE My_Database;

-- Store results in a local temp table so that.  I'm using a
-- local temp table so that I can access it in SP_EXECUTESQL.
create table #tmp (
    tbl nvarchar(max),
    col nvarchar(max),
    val nvarchar(max)   
);

declare @tbl nvarchar(max);
declare @col nvarchar(max);
declare @q nvarchar(max);
declare @search nvarchar(max) = 'my search key';

-- Create a cursor on all columns in the database
declare c cursor for
SELECT tbls.TABLE_NAME, cols.COLUMN_NAME  FROM INFORMATION_SCHEMA.TABLES AS tbls
JOIN INFORMATION_SCHEMA.COLUMNS AS cols
ON tbls.TABLE_NAME = cols.TABLE_NAME

-- For each table and column pair, see if the search value exists.
open c
fetch next from c into @tbl, @col
while @@FETCH_STATUS = 0
begin
    -- Look for the search key in current table column and if found add it to the results.
    SET @q = 'INSERT INTO #tmp SELECT ''' + @tbl + ''', ''' + @col + ''', ' + @col + ' FROM ' + @tbl + ' WHERE ' + @col + ' LIKE ''%' + @search + '%'''
    EXEC SP_EXECUTESQL @q
    fetch next from c into @tbl, @col
end
close c
deallocate c

-- Get results
select * from #tmp

-- Remove local temp table.
drop table #tmp
bstricks
quelle
1

Möglicherweise müssen Sie einen invertierten Index für Ihre Datenbank erstellen . Es ist sicher, ziemlich schnell zu sein.

Zhongjun 'Mark' Jin
quelle
1

Angenommen, wenn Sie alle Tabellen mit Namen abrufen möchten, enthält ein Spaltenname die Anmeldezeit in der Datenbank. MyDatabase unten ist das Codebeispiel

    use MyDatabase

    SELECT t.name AS table_name,
    SCHEMA_NAME(schema_id) AS schema_name,
    c.name AS column_name
    FROM sys.tables AS t
    INNER JOIN sys.columns c ON t.OBJECT_ID = c.OBJECT_ID
    WHERE c.name LIKE '%logintime%'
    ORDER BY schema_name, table_name;
Rinoy Ashokan
quelle
1

Wenn Sie phpMyAdmin installiert haben, verwenden Sie die Suchfunktion .

Wählen Sie Ihre Datenbank aus.

Stellen Sie sicher, dass Sie Datenbank und keine Tabelle ausgewählt haben. Andernfalls wird ein völlig anderer Suchdialog angezeigt.

  1. Klicken Sie auf die Registerkarte Suchen
  2. Listenelement Wählen Sie den gewünschten Suchbegriff
  3. Wählen Sie die zu durchsuchenden Tabellen aus
Anil Gupta
quelle
0
-- exec pSearchAllTables 'M54*'

ALTER PROC pSearchAllTables (@SearchStr NVARCHAR(100))
AS
BEGIN
    -- A procedure to search all tables in a database for a value
    -- Note: Use * or % for wildcard

    DECLARE 
        @Results TABLE([Schema.Table.ColumnName] NVARCHAR(370), ColumnValue NVARCHAR(3630))

    SET NOCOUNT ON

    DECLARE 
        @TableName NVARCHAR(256) = ''
        , @ColumnName NVARCHAR(128)     
        , @SearchStr2 NVARCHAR(110) = QUOTENAME(REPLACE(@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')
                        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 
        [Schema.Table.ColumnName]
        , ColumnValue 
    FROM @Results
    GROUP BY 
        [Schema.Table.ColumnName]
        , ColumnValue 

END
WonderWorker
quelle
Mir ist nicht klar, wie das funktioniert. Es sieht so aus, als gäbe es 2 Suchzeichenfolgen oder eine Such- und Ersetzungszeichenfolge? Wenn ich nur suchen möchte, wo platziere ich die gesuchte Zeichenfolge?
SherylHohman
0

Zu Entwicklungszwecken können Sie einfach die erforderlichen Tabellendaten in einen einzigen HTML-Code exportieren und direkt danach suchen.

Jeff Ayan
quelle
0

Datenbank-Client-Tools (wie DBeaver oder phpMyAdmin ) unterstützen häufig die Volltextsuche in der gesamten Datenbank.

Yuri
quelle