Как передать список, разделенный запятыми, в хранимую процедуру?
-
08-06-2019 - |
Вопрос
Итак, у меня есть хранимый в 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
Решение
Если вы используете Sybase 12.5 или более раннюю версию, то вы не можете использовать функции.Обходным решением может быть заполнение временной таблицы значениями и чтение их оттуда.
Другие советы
Это немного запоздало, но некоторое время назад у меня была именно эта проблема, и я нашел решение.
Хитрость заключается в двойных кавычках, а затем в том, чтобы заключить всю строку в кавычки.
exec getSomething """John"",""Tom"",""Bob"",""Harry"""
Измените свой процесс так, чтобы запись в таблице соответствовала строке.
CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE @keyList LIKE '%'+name+'%'
У меня это было в производстве с ASE 12.5;сейчас мы переходим к 15.0.3.
Передайте список, разделенный запятыми, в функцию, которая возвращает табличное значение.Где-то в 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 должна быть способна сделать что-то подобное?
Вам нужно использовать список, разделенный запятыми?Последние пару лет я использовал идею такого рода и передавал ее в XML-файле."Функция" openxml принимает строку и преобразует ее в xml, а затем, если вы создаете временную таблицу с данными, к ней можно запрашивать.
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))
Что касается идеи Кевина передать параметр функции, которая разбивает текст на таблицы, вот моя реализация этой функции, созданная несколько лет назад.Работает с удовольствием.
Это быстрый и грязный метод, который может быть полезен:
select *
from mytbl
where "," + ltrim(rtrim(@keylist)) + "," like "%," + ltrim(rtrim(name)) + ",%"
Не уверен, что это в ASE, но в SQL Anywhere sa_split_list список sa_split_list функция возвращает таблицу из CSV-файла.У него есть необязательные аргументы для передачи другого разделителя (по умолчанию используется запятая) и максимальной длины для каждого возвращаемого значения.
Проблема с подобными вызовами:exec получает что-то вроде "John", "Tom"" заключается в том, что он обрабатывает "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 заключены в кавычки, даже если они являются одиночными значениями.
Чтобы коснуться того, что предоставил @Abel, то, что помогло мне, было:
Моей целью было взять все, что когда-либо вводил конечный пользователь из SSRS, и использовать это в моем предложении where как In (SELECT) Очевидно, что @ICD_VALUE_RPT будет закомментирован в моем запросе Dataset.
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))
)
Попробуйте этот способ.У меня это работает.
@itemIds varchar(max)
CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE name IN (SELECT Value FROM [Global_Split] (@itemIds,','))
Это работает в 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)