2008-08-08 121 views
16

所以我有一个的Sybase存储过程,在一个IN()子句需要1个参数,这是一个逗号分隔的字符串列表,并运行一个查询在:如何将逗号分隔列表传递给存储过程?

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拥有的各种方式有很大的参考,以考绩阵列,以一个存储过程

+0

我希望你找到了适合你的方法。 链接的页面是一个很好的选项列表,但我很高兴看到大多数已经在这里建议! 保罗建议方法2/3,临时表。 我建议方法1,动态sql。 Brian和Abel提出了一种XML方法,虽然我没有在sybase中获得xml的许可,所以不知道这是否能在Sybase中工作。类似于方法4. – AdamH 2009-08-21 13:17:35

+0

可能重复[参数化SQL IN子句?](http://stackoverflow.com/questions/337704/parameterizing-an-sql-in-条款) – icc97 2014-02-04 15:19:53

回答

3

如果您正在使用Sybase 12.5或更早的版本,那么你不能使用的功能。解决方法可能是用值填充临时表并从中读取它们。

0

你需要使用逗号分隔列表?最近几年,我一直在采取这种类型的想法并传入一个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)) 
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应该能够做同样的事情?

0

关于凯文的想法是将参数传递给一个将文本分割成表格的函数,下面是几年前我实现该函数的一个函数。作品一种享受。

Splitting Text into Words in SQL

0

这是一个快速和肮脏的方法可能有用:

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

不知道,如果它在ASE,但在SQL Anywhere中,sa_split_list功能从CSV返回一个表。它有可选参数为每个返回的值传递不同的分隔符(默认值为逗号)和最大长度。

sa_split_list function

0

,像这样的电话的问题:EXEC getSomething“‘约翰’,‘汤姆’”为它的治疗“‘约翰’,‘汤姆’”作为一个字符串,它只会匹配表中的条目是'John','Tom''。

如果你不想在Paul的答案中使用临时表,那么你可以使用动态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

这适用于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申报与调用存储过程之前初始化XML变量的SQL代码内:

DECLARE @NameArray XML

SET @NameArray ='< ID> NAME_1 </ID> < ID> NAME_2 </ID> < ID> NAME_3 </ID> < ID> NAME_4 </ID>”

使用您的例子在调用存储过程将是:

EXEC GetSomething @NameArray

我以前也用这个方法,它工作正常。如果你想快速测试,复制下面的代码粘贴到一个新的查询,并执行:

DECLARE @IdArray XML

SET @IdArray ='< ID> NAME_1 </ID> < ID > NAME_2 </ID> < ID> NAME_3 </ID> < ID> NAME_4 </ID> '

SELECT ParamValues.ID.value('。 '' VARCHAR(10)') FROM @IdArray。节点('id')AS ParamValues(ID)

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。

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

尝试这种方式。它的作品适合我。

@itemIds varchar(max) 

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