カンマ区切りのリストをストアド プロシージャに渡すにはどうすればよいですか?
-
08-06-2019 - |
質問
そこで、カンマ区切りの文字列リストである 1 つのパラメータを受け取り、IN() 句で in を使用してクエリを実行する Sybase ストアド プロシージャを用意しました。
CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE name IN (@keyList)
リストに複数の値を指定してストアド プロシージャを呼び出すにはどうすればよいですか?これまで私が試してきたこと
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 以前を使用している場合は、関数を使用できません。回避策としては、一時テーブルに値を設定し、そこから値を読み取ることが考えられます。
他のヒント
少し遅くなりましたが、少し前にまさにこの問題が発生し、解決策を見つけました。
秘訣は、二重引用符で囲み、文字列全体を引用符で囲むことです。
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 です。
カンマ区切りのリストをテーブル値を返す関数に渡します。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 も同様のことができるはずだと思いますか?
カンマ区切りリストを使用する必要がありますか?ここ数年、私はこの種のアイデアを採用し、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))
テキストをテーブルに分割する関数にパラメータを渡すという Kevin のアイデアに関して、数年前の私によるその関数の実装を次に示します。ご馳走になります。
これは、便利かもしれない手っ取り早い方法です。
select *
from mytbl
where "," + ltrim(rtrim(@keylist)) + "," like "%," + ltrim(rtrim(name)) + ",%"
ASE かどうかはわかりませんが、SQL Anywhere では、 sa_split_list 関数は CSV からテーブルを返します。異なる区切り文字 (デフォルトはカンマ) と各戻り値の最大長を渡すためのオプションの引数があります。
次のような呼び出しの問題:exec getSomething '"John","Tom"' は、'"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 内の項目は、単一の値であっても引用符で囲む必要があります。
@Abel が提供してくれたものに触れると、私にとって役に立ったのは次のとおりです。
私の目的は、エンドユーザーがSSRSから入力したものを取り、私の句でそれを(select)として使用することでした。
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))
)
この方法を試してください。それは私にとってはうまくいきます。
@itemIds varchar(max)
CREATE PROCEDURE getSomething @keyList varchar(4096)
AS
SELECT * FROM mytbl WHERE name IN (SELECT Value FROM [Global_Split] (@itemIds,','))
これは 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)