incluindo parâmetros em OPENQUERY

Como posso usar um parâmetro dentro do openquery sql, como:

SELECT * FROM OPENQUERY([NameOfLinkedSERVER], 'SELECT * FROM TABLENAME
where field1=@someParameter') T1 INNER JOIN MYSQLSERVER.DATABASE.DBO.TABLENAME
T2 ON T1.PK = T2.PK
Author: Sarfraz, 2010-07-31

13 answers

Da documentação diz que:

O OPENQUERY não aceita variáveis pelos seus argumentos.

Ver este artigo para uma solução.

Actualizar:

Como sugerido, estou a incluir as recomendações do artigo abaixo.

Passar Os Valores Básicos

Quando a instrução básica de Transact-SQL é conhecida, mas você tem que passar em um ou mais valores específicos, use o código que é semelhante à seguinte amostra:

DECLARE @TSQL varchar(8000), @VAR char(2)
SELECT  @VAR = 'CA'
SELECT  @TSQL = 'SELECT * FROM OPENQUERY(MyLinkedServer,''SELECT * FROM pubs.dbo.authors WHERE state = ''''' + @VAR + ''''''')'
EXEC (@TSQL)

Passar a pesquisa inteira

Quando tiver de passar em toda a consulta Transact-SQL ou o nome do servidor ligado( ou ambos), use um código semelhante à seguinte amostra:

DECLARE @OPENQUERY nvarchar(4000), @TSQL nvarchar(4000), @LinkedServer nvarchar(4000)
SET @LinkedServer = 'MyLinkedServer'
SET @OPENQUERY = 'SELECT * FROM OPENQUERY('+ @LinkedServer + ','''
SET @TSQL = 'SELECT au_lname, au_id FROM pubs..authors'')' 
EXEC (@OPENQUERY+@TSQL) 

Usar o procedimento de armazenamento Sp_ ExecuteSQL

Para evitar as aspas multi-camadas, use um código semelhante à seguinte amostra:

DECLARE @VAR char(2)
SELECT  @VAR = 'CA'
EXEC MyLinkedServer.master.dbo.sp_executesql
N'SELECT * FROM pubs.dbo.authors WHERE state = @state',
N'@state char(2)',
@VAR
 156
Author: Garett, 2014-11-10 16:53:09
Pode executar um texto com OPENQUERY quando o construir. Se você for por este caminho, pense na segurança e tenha cuidado para não concatenar o texto inserido pelo usuário em seu SQL!
DECLARE @Sql VARCHAR(8000)
SET @Sql = 'SELECT * FROM Tbl WHERE Field1 < ''someVal'' AND Field2 IN '+ @valueList 
SET @Sql = 'SELECT * FROM OPENQUERY(SVRNAME, ''' + REPLACE(@Sql, '''', '''''') + ''')'
EXEC(@Sql)
 15
Author: Tahbaza, 2010-07-31 14:47:24

Da Página MSDN:

O OPENQUERY não aceita variáveis para os seus argumentos

Fundamentalmente, isso significa que você não pode emitir uma consulta dinâmica. Para alcançar o que a sua amostra está tentando, tente isto:
SELECT * FROM 
   OPENQUERY([NameOfLinkedSERVER], 'SELECT * FROM TABLENAME') T1 
   INNER JOIN 
   MYSQLSERVER.DATABASE.DBO.TABLENAME T2 ON T1.PK = T2.PK 
where
   T1.field1 = @someParameter

Claramente se a sua tabela de nomes de tabela contém uma grande quantidade de dados, Isto irá atravessar a rede também e o desempenho pode ser pobre. Por outro lado, para uma pequena quantidade de dados, isso funciona bem e evita a construção dinâmica sql despesas gerais (injecção sql, cotações de escape) que uma abordagem exec pode exigir.

 13
Author: Neil Moss, 2010-07-31 14:56:00
Na verdade, encontrámos uma maneira de fazer isto.
DECLARE @username varchar(50)
SET @username = 'username'
DECLARE @Output as numeric(18,4)
DECLARE @OpenSelect As nvarchar(500)
SET @OpenSelect = '(SELECT @Output = CAST((CAST(pwdLastSet As bigint) / 864000000000) As numeric(18,4)) FROM OpenQuery (ADSI,''SELECT pwdLastSet
                                FROM  ''''LDAP://domain.net.intra/DC=domain,DC=net,DC=intra''''
                                WHERE objectClass =  ''''User'''' AND sAMAccountName = ''''' + @username + '''''
                          '') AS tblADSI)'
EXEC sp_executesql @OpenSelect, N'@Output numeric(18,4) out', @Output out
SELECT @Output As Outputs

Isto irá atribuir o resultado da execução do OpenQuery, na variável @Output.

Testamos o procedimento de armazenamento no MSSQL 2012, mas devemos trabalhar com o MSSQL 2008+.

A Microsoft diz que o sp_executesql (Transact-SQL): aplica-se a: SQL Server (SQL Server 2008 através da versão actual), Windows Azure SQL Database (Versão inicial através da versão actual). ([9]} http://msdn.microsoft.com/en-us/library/ms188001.aspx)

 7
Author: Juan Medina, 2014-06-12 22:21:08
DECLARE @guid varchar(36);  select @guid= convert(varchar(36), NEWID() );
/*
    The one caveat to this technique is that ##ContextSpecificGlobal__Temp should ALWAYS have the exact same columns.  
    So make up your global temp table name in the sproc you're using it in and only there!
    In this example I wanted to pass in the name of a global temporary table dynamically.  I have 1 procedure dropping 
    off temporary data in whatever @TableSrc is and another procedure picking it up but we are dynamically passing 
    in the name of our pickup table as a parameter for OPENQUERY.
*/
IF ( OBJECT_ID('tempdb..##ContextSpecificGlobal__Temp' , 'U') IS NULL )
    EXEC ('SELECT * INTO ##ContextSpecificGlobal__Temp FROM OPENQUERY(loopback, ''Select *,''''' +  @guid +''''' as tempid FROM ' + @TableSrc + ''')')
ELSE 
    EXEC ('INSERT ##ContextSpecificGlobal__Temp SELECT * FROM OPENQUERY(loopback, ''Select *,''''' +  @guid +''''' as tempid FROM ' + @TableSrc + ''')')

--If this proc is run frequently we could run into race conditions, that's why we are adding a guid and only deleting
--the data we added to ##ContextSpecificGlobal__Temp
SELECT * INTO #TableSrc FROM ##ContextSpecificGlobal__Temp WHERE tempid = @guid

BEGIN TRAN t1
    IF ( OBJECT_ID('tempdb..##ContextSpecificGlobal__Temp' , 'U') IS NOT NULL ) 
    BEGIN
        -- Here we wipe out our left overs if there if everyones done eating the data
        IF (SELECT COUNT(*) FROM ##ContextSpecificGlobal__Temp) = 0
            DROP TABLE ##ContextSpecificGlobal__Temp
    END
COMMIT TRAN t1

-- YEAH! Now I can use the data from my openquery without wrapping the whole !$#@$@ thing in a string.
 4
Author: Ryan Maloney, 2012-05-04 16:28:00
SELECT field1 FROM OPENQUERY 
                   ([NameOfLinkedSERVER], 
                   'SELECT field1 FROM TABLENAME') 
                           WHERE field1=@someParameter T1 
                                 INNER JOIN MYSQLSERVER.DATABASE.DBO.TABLENAME           
                                 T2 ON T1.PK = T2.PK
 2
Author: Tuan Zaidi, 2016-09-14 09:31:56

Combinar SQL dinâmico com OpenQuery. (Isto vai para um servidor Teradata)

DECLARE 
    @dayOfWk    TINYINT = DATEPART(DW, GETDATE()),
    @qSQL       NVARCHAR(MAX) = '';

SET @qSQL = '
SELECT
    *
FROM
    OPENQUERY(TERASERVER,''
        SELECT DISTINCT
            CASE
                WHEN ' + CAST(@dayOfWk AS NCHAR(1)) + ' = 2
                THEN ''''Monday''''
                ELSE ''''Not Monday''''
            END
        '');';

EXEC sp_executesql @qSQL;
 2
Author: Mike, 2017-08-16 15:02:34

No exemplo seguinte estou a passar um parâmetro do departamento para um procedimento armazenado(spaumentasetotalsrpt) e ao mesmo tempo estou a criar uma tabela temporária tudo a partir de um OPENQUERY. A tabela Temp precisa ser uma temperatura global ( # # ) para que possa ser referenciada fora da sua intância. Ao usar exec sp_ ExecuteSQL você pode passar o parâmetro do Departamento.

Nota: Tenha cuidado ao utilizar sp_ ExecuteSQL. Também o seu administrador pode não ter esta opção disponível para você.

Espero que isto ajude alguém.
 IF OBJECT_ID('tempdb..##Temp') IS NOT NULL
/*Then it exists*/
    begin
       DROP TABLE ##Temp
    end 
 Declare @Dept as nvarchar(20) ='''47'''

 declare @OPENQUERY  as nvarchar(max)
set @OPENQUERY = 'Select ' + @Dept + ' AS Dept,  * into ##Temp from openquery(SQL_AWSPROD01,''' 

declare @sql nvarchar(max)= @openquery +  'SET FMTONLY OFF EXECUTE SalaryCompensation.dbo.spIncreaseTotalsRpts ' + '''' + @Dept + ''''  + ''')'
declare @parmdef nvarchar(25) 
DECLARE @param nvarchar(20) 

SET @parmdef = N'@Dept varchar(20)'
-- select @sql
-- Print @sql + @parmdef  + @dept
exec sp_executesql @sql,@parmdef, @Dept  
Select * from ##Temp

Resultados

Aumento de Dept CNT 0 1 2 3 4 5 6 0.0000 1.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000

 1
Author: Carlos, 2015-02-05 10:32:56
Descobri uma maneira que funciona para mim. Ele requer o uso de uma tabela de arranhões a que um servidor ligado tem acesso.

Criei uma tabela e preenchi-a com os valores de que preciso, depois referencio essa tabela através de um servidor ligado.

SELECT * 
FROM OPENQUERY(KHSSQLODSPRD,'SELECT *
  FROM ABC.dbo.CLAIM A WITH (NOLOCK)
  WHERE A.DOS >= (SELECT MAX(DATE) FROM KHSDASQL01.DA_MAIN.[dbo].[ALLFILENAMES]) ')
 0
Author: Hannover Fist, 2015-08-12 23:40:54
declare @p_Id varchar(10)
SET @p_Id = '40381'

EXECUTE ('BEGIN update TableName
                set     ColumnName1 = null,
                        ColumnName2 = null,
                        ColumnName3 = null,
                        ColumnName4 = null
                 where   PERSONID = '+ @p_Id +'; END;') AT [linked_Server_Name]
 0
Author: Sachin Khartode, 2015-08-19 10:01:25

Podemos usar o método execute em vez de openquery. O seu código é muito mais limpo. Tive de obter o resultado da consulta numa variável. Usei o seguinte código.

CREATE TABLE #selected_store
(
   code VARCHAR(250),
   id INT
)
declare @storeId as integer = 25
insert into #selected_store (id, code) execute('SELECT store_id, code from quickstartproductionnew.store where store_id = ?', @storeId) at [MYSQL]  

declare @code as varchar(100)
select @code = code from #selected_store
select @code
drop table #selected_store

Nota:

Se a sua pesquisa não funcionar, certifique-se que remote proc transaction promotion está definido como false para a sua ligação linked server.

EXEC master.dbo.sp_serveroption
       @server = N'{linked server name}',
       @optname = N'remote proc transaction promotion',
       @optvalue = N'false';
 0
Author: Sheikh Abdul Wahid, 2020-04-23 09:54:26

Exemplo simples baseado no exemplo de @Tuan Zaidi acima do qual parecia o mais fácil. Não sabia que conseguias fazer o filtro do lado de fora do OPENQUERY... muito mais fácil!

No entanto, no meu caso, eu precisava colocá-lo em uma variável, então eu criei um nível de sub consulta adicional para retornar um único valor.

SET @SFID = (SELECT T.Id FROM (SELECT Id,  Contact_ID_SQL__c  FROM OPENQUERY([TR-SF-PROD], 'SELECT Id,  Contact_ID_SQL__c FROM Contact') WHERE Contact_ID_SQL__c = @ContactID) T)
 -1
Author: Anthony Griggs, 2016-09-27 19:15:56
Tente assim, deve funcionar, com calma! Na sua cláusula onde, após o nome da coluna e igual a assinar: - adicione duas aspas simples, o seu valor de pesquisa e, em seguida, três aspas simples. Fecha o suporte.

SELECT * FROM OPENQUERY([NameOfLinkedSERVER], 'SELECT * FROM TABLENAME where field1=''your search value''') T1 INNER JOIN MYSQLSERVER.DATABASE.DBO.TABLENAME T2 ON T1.PK = T2.PK

 -1
Author: ParagDI, 2020-11-19 21:06:09