如何将逗号分隔列表传递给存储过程?

时间:2008-08-08 20:48:29

标签: sql sybase-ase

所以我有一个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

编辑:我实际上发现了这个page,它很好地引用了将数组传递给sproc的各种方法

11 个答案:

答案 0 :(得分:3)

如果您使用的是Sybase 12.5或更早版本,则无法使用函数。解决方法可能是使用值填充临时表并从那里读取它们。

答案 1 :(得分:2)

这有点晚了,但我刚才有这个确切的问题,我找到了一个解决方案。

诀窍是双引号,然后将整个字符串包装在引号中。

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。

答案 2 :(得分:1)

将逗号分隔列表传递给返回表值的函数。 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应该可以做类似的事情吗?

答案 3 :(得分:0)

您需要使用以逗号分隔的列表吗?在过去的几年里,我一直在采用这种想法并传入一个XML文件。 openxml“function”接受一个字符串并使其像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))

答案 4 :(得分:0)

关于Kevin将参数传递给将文本拆分为表的函数的想法,这是几年前我对该函数的实现。是一种享受。

Splitting Text into Words in SQL

答案 5 :(得分:0)

这是一种快速而肮脏的方法,可能很有用:

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

答案 6 :(得分:0)

不确定它是否在ASE中,但在SQL Anywhere中, sa_split_list 函数从CSV返回一个表。它有可选的参数来传递不同的分隔符(默认是逗号)和每个返回值的maxlength。

sa_split_list function

答案 7 :(得分:0)

这样的调用的问题: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中的项目有引号,即使它们是单个值。

答案 8 :(得分:0)

这适用于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))

从调用SP的SQL代码中,在调用存储过程之前声明并初始化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)

答案 9 :(得分:0)

要了解@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))
)

答案 10 :(得分:0)

试试这种方式。它适用于我。

@itemIds varchar(max)

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