关于sql:如何将逗号分隔的列表传递给存储过程?

关于sql:如何将逗号分隔的列表传递给存储过程?

How to pass a comma separated list to a stored procedure?

因此,我有一个Sybase存储的proc,它采用1个参数,该参数是逗号分隔的字符串列表,并在IN()子句中使用in运行查询:

1
2
3
CREATE PROCEDURE getSomething @keyList VARCHAR(4096)
AS
SELECT * FROM mytbl WHERE name IN (@keyList)

如何在列表中调用具有大于1的值的存储过程?
到目前为止,我已经尝试过

1
2
3
4
5
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或更早版本,则不能使用函数。一种解决方法是用值填充临时表并从中读取它们。


这有点晚了,但是前一阵子我遇到了这个确切的问题,我找到了解决方案。

技巧是双引号,然后将整个字符串用引号引起来。

1
EXEC getSomething"""John"",""Tom"",""Bob"",""Harry"""

修改您的proc以使表条目与字符串匹配。

1
2
3
CREATE PROCEDURE getSomething @keyList VARCHAR(4096)
AS
SELECT * FROM mytbl WHERE @keyList LIKE '%'+name+'%'

自ASE 12.5起,我就已经开始生产此产品。我们现在是15.0.3。


将以逗号分隔的列表传递给返回表值的函数。在StackOverflow上的某个地方有一个MS SQL示例,如果我现在可以看到它,该死的。

1
2
3
CREATE PROCEDURE getSomething @keyList VARCHAR(4096)
AS
SELECT * FROM mytbl WHERE name IN (fn_GetKeyList(@keyList))

致电-

1
EXEC getSomething 'John,Tom,Foo,Bar'

我猜测Sybase应该能够执行类似的操作?


尝试这种方式。它对我有用。

1
2
3
4
5
@itemIds VARCHAR(MAX)

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

谈到@Abel提供的内容,帮助我的是:

我的目的是接受最终用户从SSRS输入的内容,并在我的where子句中将其用作In(SELECT)
显然,@ ICD_VALUE_RPT将在我的数据集查询中被注释掉。

1
2
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中添加:

1
2
3
(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))
)

这在SQL中有效。在您的GetSomething过程中声明一个XML类型的变量,如下所示:

1
DECLARE @NameArray XML = NULL

存储过程的主体实现以下内容:

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

在调用SP的SQL代码中,调用SP来声明和初始化XML变量,然后再调用存储过程:

1
2
3
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>'

使用您的示例,对存储过程的调用将是:

1
EXEC GetSomething @NameArray

我以前使用过这种方法,并且效果很好。如果要进行快速测试,请将以下代码复制并粘贴到新查询中并执行:

1
2
3
4
5
6
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)

这样的调用存在问题:exec getSomething'" John"," Tom"'是将'" John"," Tom"'视为单个字符串,它将仅匹配表中的条目""约翰","汤姆"。

如果您不想像Paul的回答那样使用临时表,则可以使用动态sql。 (假设v12 +)

1
2
3
4
5
CREATE PROCEDURE getSomething @keyList VARCHAR(4096)
AS
DECLARE @SQL VARCHAR(4096)
SELECT @SQL ="SELECT * FROM mytbl WHERE name IN (" + @keyList +")"
EXEC(@SQL)

您将需要确保@keylist中的项目用引号引起来,即使它们是单个值也是如此。


不知道它是否在ASE中,但是在SQL Anywhere中,sa_split_list函数从CSV返回一个表。它具有可选参数,以传递不同的定界符(默认为逗号)和每个返回值的最大长度。

sa_split_list函数


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

1
2
3
SELECT  *
FROM    mytbl
WHERE  "," + ltrim(rtrim(@keylist)) +"," LIKE"%," + ltrim(rtrim(name)) +",%"

关于Kevin将参数传递给将文本分割成表格的函数的想法,这是几年前我对该函数的实现。工作请客。

在SQL中将文本拆分成单词


您是否需要使用逗号分隔的列表?在过去的几年中,我一直采用这种想法并传递XML文件。 openxml"函数"采用字符串并将其设置为类似于xml的字符串,然后如果您使用数据创建临时表,则该表是可查询的。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
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))


推荐阅读