Как передать список через запятую хранимой процедуре? - PullRequest
17 голосов
/ 09 августа 2008

Итак, у меня есть сохраненный процесс Sybase, который принимает 1 параметр, который представляет собой список строк, разделенных запятыми, и выполняет запрос с параметром in в условии IN ():

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

Как мне вызвать мой сохраненный процесс с более чем 1 значением в списке? Пока я пробовал

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

РЕДАКТИРОВАТЬ: На самом деле я нашел эту страницу , в которой есть множество ссылок на различные способы передачи массива в sproc

Ответы [ 11 ]

3 голосов
/ 14 августа 2008

Если вы используете Sybase 12.5 или более раннюю версию, вы не можете использовать функции. Обходным путем может быть заполнение временной таблицы значениями и чтение их оттуда.

2 голосов
/ 25 октября 2011

Это немного поздно, но у меня была именно эта проблема некоторое время назад, и я нашел решение.

Уловка заключается в двойных кавычках, а затем заключает в кавычки всю строку.

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

Измените ваш proc, чтобы он соответствовал записи таблицы в строке.

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

У меня это было в производстве с ASE 12.5; мы сейчас на 15.0.3.

1 голос
/ 09 августа 2008

Передать список через запятую в функцию, которая возвращает табличное значение. Где-то на StackOverflow есть пример MS SQL, черт возьми, если я его сейчас вижу.

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

Позвонить с -

exec getSomething 'John,Tom,Foo,Bar'

Я предполагаю, что Sybase сможет сделать что-то подобное?

0 голосов
/ 07 сентября 2017

Попробуйте так. Это работает для меня.

@itemIds varchar(max)

CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE name IN (SELECT Value FROM [Global_Split] (@itemIds,','))
0 голосов
/ 03 июня 2016

Чтобы коснуться того, что предоставил @Abel, мне помогло:

Моя цель состояла в том, чтобы взять то, что конечный пользователь ввел из SSRS, и использовать это в предложении where как In (SELECT) Очевидно, что @ICD_VALUE_RPT будет закомментировано в моем запросе к набору данных.

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>')

тогда в моем WHERE я добавил:

(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))
)
0 голосов
/ 30 октября 2008

Это работает в SQL. Объявите в вашей GetSomething процедуре переменную типа XML следующим образом:

DECLARE @NameArray XML = NULL

В теле хранимой процедуры реализовано следующее:

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

Из кода SQL, который вызывает SP для объявления и инициализации переменной XML перед вызовом хранимой процедуры:

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>'

Используя ваш пример, вызов хранимой процедуры будет:

EXEC GetSomething @NameArray

Я использовал этот метод раньше, и он отлично работает. Если вам нужен быстрый тест, скопируйте и вставьте следующий код в новый запрос и выполните:

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)
0 голосов
/ 24 сентября 2008

Проблема с такими вызовами: exec getSomething '' John ',' Tom '' состоит в том, что он обрабатывает '' John ',' Tom '' как одну строку, он будет соответствовать только записи в таблице это «Джон», «Том».

Если вы не хотите использовать временную таблицу, как в ответе Пола, тогда вы можете использовать динамический sql. (Предполагается, v12 +)

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

Вам необходимо убедиться, что элементы в @keylist заключены в кавычки, даже если они представляют собой отдельные значения.

0 голосов
/ 24 сентября 2008

Не уверен, что он в ASE, но в SQL Anywhere функция sa_split_list возвращает таблицу из CSV. Он имеет необязательные аргументы для передачи другого разделителя (по умолчанию - запятая) и максимальной длины для каждого возвращаемого значения.

Функция sa_split_list

0 голосов
/ 07 сентября 2008

Это быстрый и грязный метод, который может быть полезен:

select  * 
from    mytbl 
where   "," + ltrim(rtrim(@keylist)) + "," like "%," + ltrim(rtrim(name)) + ",%"
0 голосов
/ 09 августа 2008

Что касается идеи Кевина о передаче параметра в функцию, которая разбивает текст на таблицу, то вот моя реализация этой функции несколько лет назад. Работает угощение.

Разделение текста на слова в SQL

...