2010-05-15 18 views
106

sto provando ad eseguire questa query:Tabella come variabile

declare @tablename varchar(50) 
set @tablename = 'test' 
select * from @tablename 

Questo produce il seguente errore:

Msg 1087, Level 16, State 1, Line 5

Must declare the table variable "@tablename".

Qual è il modo giusto per avere il nome della tabella popolato dinamicamente?

risposta

91

nomi delle tabelle e delle colonne devono essere statico, se la query è statico. Per i nomi dinamici di tabelle o colonne, è necessario generare l'SQL completo in modo dinamico e utilizzare sp_executesql per eseguirlo.

Maggiori dettagli qui: The curse and blessings of dynamic SQL

31

Non è possibile utilizzare un nome di tabella per una variabile, che avrebbe dovuto fare questo, invece:

DECLARE @sqlCommand varchar(1000) 
SET @sqlCommand = 'SELECT * from yourtable' 
EXEC (@sqlCommand) 
9

Avrai bisogno di generare l'SQL dinamico:

declare @tablename varchar(50) 

set @tablename = 'test' 

declare @sql varchar(500) 

set @sql = 'select * from ' + @tablename 

exec (@sql) 
59

Cambia la tua ultima affermazione a questo:

EXEC('SELECT * FROM ' + @tablename) 

Questo è come lo faccio il mio in una stored procedure. Il primo blocco dichiara la variabile e imposta il nome della tabella in base al nome dell'anno e del mese corrente, in questo caso TEST_2012OCTOBER. Quindi controllo se esiste già nel DB e rimuoverlo se lo fa. Quindi il prossimo blocco utilizzerà un'istruzione SELECT INTO per creare la tabella e popolarla con i record di un'altra tabella con i parametri.

--DECLARE TABLE NAME VARIABLE DYNAMICALLY 
DECLARE @table_name varchar(max) 
SET @table_name = 
    (SELECT 'TEST_' 
      + DATENAME(YEAR,GETDATE()) 
      + UPPER(DATENAME(MONTH,GETDATE()))) 

--DROP THE TABLE IF IT ALREADY EXISTS 
IF EXISTS(SELECT name 
      FROM sysobjects 
      WHERE name = @table_name AND xtype = 'U') 

BEGIN 
    EXEC('drop table ' + @table_name) 
END 

--CREATES TABLE FROM DYNAMIC VARIABLE AND INSERTS ROWS FROM ANOTHER TABLE 
EXEC('SELECT * INTO ' + @table_name + ' FROM dbo.MASTER WHERE STATUS_CD = ''A''') 
26

po 'tardi per una risposta, ma dovrebbe aiutare qualcun altro:

CREATE PROCEDURE [dbo].[GetByName] 
    @TableName NVARCHAR(100) 
    AS 
BEGIN 
    -- SET NOCOUNT ON added to prevent extra result sets from 
    -- interfering with SELECT statements. 
    SET NOCOUNT ON; 
    DECLARE @sSQL nvarchar(500); 

    SELECT @sSQL = N'SELECT * FROM' + QUOTENAME(@TableName); 

    EXEC sp_executesql @sSQL 



END 
+6

QUOTENAME è importante per la sicurezza. Grazie. –

+0

Ma come restituire il valore da tale query? Per esempio. 'COUNT (*)'? – Suncatcher

-1
Declare @fs_e int, @C_Tables CURSOR, @Table varchar(50) 

SET @C_Tables = CURSOR FOR 
     select name from sysobjects where OBJECTPROPERTY(id, N'IsUserTable') = 1 AND name like 'TR_%' 
OPEN @C_Tables 
FETCH @C_Tables INTO @Table 
    SELECT @fs_e = sdec.fetch_Status FROM sys.dm_exec_cursors(0) as sdec where sdec.name = '@C_Tables' 

WHILE (@fs_e <> -1) 
    BEGIN 
     exec('Select * from '+ @Table) 
     FETCH @C_Tables INTO @Table 
     SELECT @fs_e = sdec.fetch_Status FROM sys.dm_exec_cursors(0) as sdec where sdec.name = '@C_Tables' 
    END 
1
DECLARE @tbl sysname, 
     @sql nvarchar(4000), 
     @params nvarchar(4000), 
     @count int 

DECLARE tblcur CURSOR STATIC LOCAL FOR 
    SELECT object_name(id) FROM syscolumns WHERE name = 'LastUpdated' 
    ORDER BY 1 
OPEN tblcur 

WHILE 1 = 1 
BEGIN 
    FETCH tblcur INTO @tbl 
    IF @@fetch_status <> 0 
     BREAK 

    SELECT @sql = 
    N' SELECT @cnt = COUNT(*) FROM dbo.' + quotename(@tbl) + 
    N' WHERE LastUpdated BETWEEN @fromdate AND ' + 
    N'       coalesce(@todate, ''99991231'')' 
    SELECT @params = N'@fromdate datetime, ' + 
        N'@todate datetime = NULL, ' + 
        N'@cnt  int  OUTPUT' 
    EXEC sp_executesql @sql, @params, '20060101', @cnt = @count OUTPUT 

    PRINT @tbl + ': ' + convert(varchar(10), @count) + ' modified rows.' 
END 

DEALLOCATE tblcur 

ho messo le linee che p

+0

Questo post deve essere modificato e potrebbe essere necessario qualche spiegazione .. –