19

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

编辑:我实际上发现这个页面对将数组传递给存储过程的各种方法有很好的参考

4

11 回答 11

4

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

于 2008-08-14T09:27:30.793 回答
2

这有点晚了,但不久前我遇到了这个确切的问题,我找到了解决方案。

诀窍是双引号,然后将整个字符串用引号括起来。

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。

于 2011-10-25T16:01:30.880 回答
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 应该能够做类似的事情?

于 2008-08-08T21:04:58.703 回答
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))
于 2008-08-08T20:57:51.150 回答
0

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

在 SQL 中将文本拆分为单词

于 2008-08-09T00:05:04.847 回答
0

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

select  * 
from    mytbl 
where   "," + ltrim(rtrim(@keylist)) + "," like "%," + ltrim(rtrim(name)) + ",%"
于 2008-09-07T13:18:46.057 回答
0

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

sa_split_list 函数

于 2008-09-23T20:17:38.717 回答
0

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

于 2008-09-23T20:36:39.560 回答
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><</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)
于 2008-10-30T17:57:16.700 回答
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))
)
于 2016-06-03T01:13:32.650 回答
0

试试这个方法。它对我有用。

@itemIds varchar(max)

CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE name IN (SELECT Value FROM [Global_Split] (@itemIds,','))
于 2017-09-07T09:06:03.437 回答