如何将逗号分隔的列表传递给存储过程?
所以我有一个Sybase存储的proc,它带有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的各种方法有很好的参考
解决方案
回答
我们是否需要使用逗号分隔的列表?最近几年,我一直采用这种想法,并传递一个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))
回答
将逗号分隔的列表传递到返回表值的函数中。在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应该能够执行类似的操作?
回答
关于Kevin将参数传递给将文本分割成表格的函数的想法,这是几年前我对该函数的实现。工作请客。
在SQL中将文本拆分为单词
回答
如果使用Sybase 12.5或者更早版本,则不能使用函数。一种解决方法是用值填充临时表并从中读取它们。
回答
这是一种快速而肮脏的方法,可能会有用:
select * from mytbl where "," + ltrim(rtrim(@keylist)) + "," like "%," + ltrim(rtrim(name)) + ",%"
回答
不知道它是否在ASE中,但是在SQL Anywhere中,sa_split_list函数从CSV返回一个表。它具有可选参数来传递不同的定界符(默认为逗号)和每个返回值的最大长度。
sa_split_list函数
回答
这样的调用存在问题:exec getSomething'" John"," Tom"'是将'" 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中的项目用引号引起来,即使它们是单个值也是如此。
回答
这在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>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.nodes('id') AS ParamValues(ID)