Como passar uma lista separada por vírgula para um procedimento armazenado?

StackOverflow https://stackoverflow.com/questions/6369

  •  08-06-2019
  •  | 
  •  

Pergunta

Então, eu tenho um processo armazenado no Sybase que usa 1 parâmetro que é uma lista de strings separadas por vírgula e executa uma consulta com in em uma cláusula IN():

CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE name IN (@keyList)

Como chamo meu processo armazenado com mais de 1 valor na lista?Até agora eu tentei

exec getSomething 'John'         -- works but only 1 value
exec getSomething 'John','Tom'   -- doesn't work - expects two variables
exec getSomething "'John','Tom'" -- doesn't work - doesn't find anything
exec getSomething '"John","Tom"' -- doesn't work - doesn't find anything
exec getSomething '\'John\',\'Tom\'' -- doesn't work - syntax error

EDITAR: Na verdade eu encontrei isso página que tem uma ótima referência das várias maneiras de passar um array para um sproc

Foi útil?

Solução

Se você estiver usando o Sybase 12.5 ou anterior, não poderá usar funções.Uma solução alternativa pode ser preencher uma tabela temporária com os valores e lê-los a partir daí.

Outras dicas

É um pouco tarde, mas tive exatamente esse problema há um tempo e encontrei uma solução.

O truque é citar duas vezes e depois colocar toda a string entre aspas.

exec getSomething """John"",""Tom"",""Bob"",""Harry"""

Modifique seu proc para corresponder a entrada da tabela à string.

CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE @keyList LIKE '%'+name+'%' 

Eu tenho isso em produção desde o ASE 12.5;estamos agora em 15.0.3.

Passe a lista separada por vírgula para uma função que retorne um valor de tabela.Há um exemplo de MS SQL em algum lugar do StackOverflow, caramba, se consigo vê-lo no momento.

CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE name IN (fn_GetKeyList(@keyList))

Ligue com -

exec getSomething 'John,Tom,Foo,Bar'

Estou supondo que o Sybase deveria ser capaz de fazer algo semelhante?

Você precisa usar uma lista separada por vírgulas?Nos últimos anos, tenho pegado esse tipo de ideia e passado um arquivo XML.A "função" openxml pega uma string e a transforma em xml e, se você criar uma tabela temporária com os dados, ela poderá ser consultada.

DECLARE @idoc int
DECLARE @doc varchar(1000)
SET @doc ='
<ROOT>
<Customer CustomerID="VINET" ContactName="Paul Henriot">
   <Order CustomerID="VINET" EmployeeID="5" OrderDate="1996-07-04T00:00:00">
      <OrderDetail OrderID="10248" ProductID="11" Quantity="12"/>
      <OrderDetail OrderID="10248" ProductID="42" Quantity="10"/>
   </Order>
</Customer>
<Customer CustomerID="LILAS" ContactName="Carlos Gonzlez">
   <Order CustomerID="LILAS" EmployeeID="3" OrderDate="1996-08-16T00:00:00">
      <OrderDetail OrderID="10283" ProductID="72" Quantity="3"/>
   </Order>
</Customer>
</ROOT>'
--Create an internal representation of the XML document.
EXEC sp_xml_preparedocument @idoc OUTPUT, @doc
-- Execute a SELECT statement that uses the OPENXML rowset provider.
SELECT    *
FROM       OPENXML (@idoc, '/ROOT/Customer',1)
            WITH (CustomerID  varchar(10),
                  ContactName varchar(20))

Com relação à ideia de Kevin de passar o parâmetro para uma função que divide o texto em uma tabela, aqui está minha implementação dessa função de alguns anos atrás.Funciona muito bem.

Dividindo texto em palavras em SQL

Este é um método rápido e sujo que pode ser útil:

select  * 
from    mytbl 
where   "," + ltrim(rtrim(@keylist)) + "," like "%," + ltrim(rtrim(name)) + ",%"

Não tenho certeza se está no ASE, mas no SQL Anywhere, o sa_split_list função retorna uma tabela de um CSV.Possui argumentos opcionais para passar um delimitador diferente (o padrão é uma vírgula) e um maxlength para cada valor retornado.

função sa_split_list

O problema com chamadas como esta:exec getSomething '"John","Tom"' é que ele está tratando '"John","Tom"' como uma única string, ele corresponderá apenas a uma entrada na tabela que é '"John","Tom"'.

Se você não quisesse usar uma tabela temporária como na resposta de Paul, poderia usar SQL dinâmico.(Assume v12+)

CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
declare @sql varchar(4096)
select @sql = "SELECT * FROM mytbl WHERE name IN (" + @keyList +")"
exec(@sql)

Você precisará garantir que os itens em @keylist tenham aspas, mesmo que sejam valores únicos.

Para falar sobre o que @Abel forneceu, o que me ajudou foi:

Meu objetivo era pegar o que o usuário final inseriu do SSRS e usá -lo na minha cláusula WHERE como uma (selecionar) obviamente @icd_value_rpt seria comentado na minha consulta de conjunto de dados.

DECLARE @ICD_VALUE_RPT VARCHAR(MAX) SET @ICD_VALUE_RPT = 'Value1, Value2'
DECLARE @ICD_VALUE_ARRAY XML SET @ICD_VALUE_ARRAY = CONCAT('<id>', REPLACE(REPLACE(@ICD_VALUE_RPT, ',', '</id>,<id>'),' ',''), '</id>')

então no meu WHERE Eu adicionei:

(PATS_WITH_PL_DIAGS.ICD10_CODE IN (SELECT ParamValues.ID.value('.','VARCHAR(MAX)') FROM @ICD_VALUE_ARRAY.nodes('id') AS ParamValues(ID))
OR PATS_WITH_PL_DIAGS.ICD9_CODE IN (SELECT ParamValues.ID.value('.','VARCHAR(MAX)') FROM @ICD_VALUE_ARRAY.nodes('id') AS ParamValues(ID))
)

Tente desta forma.Funciona para mim.

@itemIds varchar(max)

CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE name IN (SELECT Value FROM [Global_Split] (@itemIds,','))

Isso funciona em SQL.Declare em seu GetSomething processe uma variável do tipo XML como tal:

DECLARE @NameArray XML = NULL

O corpo do procedimento armazenado implementa o seguinte:

SELECT * FROM MyTbl WHERE name IN (SELECT ParamValues.ID.value('.','VARCHAR(10)')
FROM @NameArray.nodes('id') AS ParamValues(ID))

De dentro do código SQL que chama o SP para declarar e inicializar a variável XML antes de chamar o procedimento armazenado:

DECLARE @NameArray XML

SET @NameArray = '<id><</id>id>Name_1<<id>/id></id><id><</id>id>Name_2<<id>/id></id><id><</id>id>Name_3<<id>/id></id><id><</id>id>Name_4<<id>/id></id>'

Usando seu exemplo, a chamada para o procedimento armazenado seria:

EXEC GetSomething @NameArray

Eu usei esse método antes e funciona bem.Se quiser um teste rápido, copie e cole o seguinte código em uma nova consulta e execute:

DECLARE @IdArray XML

SET @IdArray = '<id><</id>id>Name_1<<id>/id></id><id><</id>id>Name_2<<id>/id></id><id><</id>id>Name_3<<id>/id></id><id><</id>id>Name_4<<id>/id></id>'

SELECT ParamValues.ID.value('.','VARCHAR(10)')
FROM @IdArray.nodes('id') AS ParamValues(ID)
Licenciado em: CC-BY-SA com atribuição
Não afiliado a StackOverflow
scroll top