sp_executesqlの結果を変数に取得する方法は?


178

実行する必要のある動的SQLがあり、その結果を変数に格納する必要があります。

私は使用できることを知っsp_executesqlていますが、これを行う方法に関する明確な例を見つけることができません。

回答:


253

あなたがすることができる出力パラメータがある場合

DECLARE @retval int   
DECLARE @sSQL nvarchar(500);
DECLARE @ParmDefinition nvarchar(500);

DECLARE @tablename nvarchar(50)  
SELECT @tablename = N'products'  

SELECT @sSQL = N'SELECT @retvalOUT = MAX(ID) FROM ' + @tablename;  
SET @ParmDefinition = N'@retvalOUT int OUTPUT';

EXEC sp_executesql @sSQL, @ParmDefinition, @retvalOUT=@retval OUTPUT;

SELECT @retval;

しかし、そうでなく、SPを変更できない場合:

-- Assuming that your SP return 1 value
create table #temptable (ID int null)
insert into #temptable exec mysp 'Value1', 'Value2'
select * from #temptable

きれいではありませんが、動作します。


私のspはsp_executesql @myQuery
JohnIdol

1
@retvalOUT=@retval OUTPUT?の3番目のパラメータはsp_executesqlちょうど@retval OUTPUTいいのではないですか?
Mohammad Dehghan 2013年

2
正接の質問です。複数のOUTPUTについてはどうですか?クエリはどのようになりますか?
Srivastav 2015

3
@SrivastavReddy私はこの答えがこれだけの賛成票を獲得する方法を得ることができません。Nishanthの答えをチェックしてください...
sotn

2
これは正しいと思います:EXEC sp_executesql @sSQL, @ParmDefinition, @retval OUTPUT;
Ian

50
DECLARE @tab AS TABLE (col1 VARCHAR(10), col2 varchar(10)) 
INSERT into @tab EXECUTE  sp_executesql N'
SELECT 1 AS col1, 2 AS col2
UNION ALL
SELECT 1 AS col1, 2 AS col2
UNION ALL
SELECT 1 AS col1, 2 AS col2'

SELECT * FROM @tab

以前にこの方法を使用したことがあります。それだけで最初に動作するようinsert into @tabです。異なるsqlでinsert into @tab複数のを実行しようとすると、最初の実行の結果のみが表示されますexecute sp_executesqlselect * from @tab
Mike Causer

おっと、私の悪い。2番目の選択でエラーが発生したため、返される行はゼロでした。この方法はうまく機能し、一時テーブルを必要としません!
マイクコーサー

これが最良の答えです。動的SQLを実行する必要がありましたが、動的SQLを最初にパラメーターに構築する必要があります。
Josh Harris

この解決策は最高の解決策です。私は他の多くの方法を試してみましたが、これが私に有効な唯一の解決策です。ニシャントありがとう
MMEL

40
DECLARE @vi INT
DECLARE @vQuery NVARCHAR(1000)

SET @vQuery = N'SELECT @vi= COUNT(*) FROM <TableName>'

EXEC SP_EXECUTESQL 
        @Query  = @vQuery
      , @Params = N'@vi INT OUTPUT'
      , @vi = @vi OUTPUT

SELECT @vi

8
変数viOUTPUTおよびviINTの宣言はどこにありますか?
Jeson Martajaya 2013

1
これは私のために働いた。投票数を増やした回答が機能しませんでした
sh_kamalh

5
パラメータ@vQueryはしなければならない NVARCHARないVARCHARとして宣言すること。
Farzad Karimi 2016


3

戻り値は通常、結果を「返す」のではなく、成功(0)またはエラー番号(1〜65K)を返すために使用されます。上記はすべて、sp_executesqlが値を返さないことを示しているようですが、これは正しくありません。sp_executesqlは、成功した場合は0を返し、失敗した場合はその他の数値を返します。

以下では、@ iは2727を返します

DECLARE @s NVARCHAR(500)
DECLARE @i INT;
SET @s = 'USE [Blah]; UPDATE STATISTICS [dbo].[TableName] [NonExistantStatisticsName];';
EXEC @i = sys.sp_executesql @s
SELECT @i AS 'Blah'

SSMSはこのメッセージ2727、レベル11、状態1、行1を表示します。インデックス 'NonExistantStaticsName'が見つかりません。


2

DECLARE @ValueTable TABLE(Value VARCHAR(100))

                    SELECT @sql = N'SELECT SRS_SizeSetDetails.'+@COLUMN_NAME+' FROM SRS_SizeSetDetails WHERE FSizeID = '''+@FSizeID+''' AND SRS_SizeSetID = '''+@SRS_SizeSetID+'''';

                    INSERT INTO @ValueTable
                    EXEC sp_executesql @sql;

                    SET @Value='';

                    SET @Value = (SELECT TOP 1  Value FROM @ValueTable)

                    DELETE FROM @ValueTable

2

複数の値を返したい場合は、次のようにします。

DECLARE @sqlstatement2      NVARCHAR(MAX);
DECLARE @retText            NVARCHAR(MAX);  
DECLARE @ParmDefinition     NVARCHAR(MAX);
DECLARE @retIndex           INT = 0;

SELECT @sqlstatement = 'SELECT @retIndexOUT=column1 @retTextOUT=column2 FROM XXX WHERE bla bla';

SET @ParmDefinition = N'@retIndexOUT INT OUTPUT, @retTextOUT NVARCHAR(MAX) OUTPUT';

exec sp_executesql @sqlstatement, @ParmDefinition, @retIndexOUT=@retIndex OUTPUT, @retTextOUT=@retText OUTPUT;

戻り値は@retIndexおよび@retTextにあります


1

ここにあなたが試すことができるものがあります

DECLARE  @SqlStatement  NVARCHAR(MAX) = ''
       ,@result     XML
       ,@DatabaseName  VARCHAR(100)
       ,@SchemaName    VARCHAR(10)
       ,@ObjectName    VARCHAR(200);

SELECT   @DatabaseName = 'some database'
       ,@SchemaName   = 'some schema'
       ,@ObjectName   = 'some object (Table/View)'

SET @SqlStatement = '
                    SELECT @result = CONVERT(XML,
                                            STUFF( ( SELECT *
                                                     FROM 
                                                       (
                                                          SELECT TOP(100) 
                                                          * 
                                                          FROM ' + QUOTENAME(@DatabaseName) +'.'+ QUOTENAME(@SchemaName) +'.' + QUOTENAME(@ObjectName) + '
                                                       ) AS A1 
                                                    FOR XML PATH(''row''), ELEMENTS, ROOT(''recordset'')
                                                 ), 1, 0, '''')
                                       )
                ';

EXEC sp_executesql @SqlStatement,N'@result XML OUTPUT', @result = @result OUTPUT;

SELECT DISTINCT
    QUOTENAME(r.value('fn:local-name(.)', 'VARCHAR(200)')) AS ColumnName
FROM @result.nodes('//recordset/*/*') AS records(r)
ORDER BY ColumnName

0

これはずっと前のことなので、まだ必要かどうかはわかりませんが、@@ ROWCOUNT変数を使用して、前のSQLステートメントで影響を受けた行数を確認できます。

これは、たとえば動的なUpdateステートメントを作成して、execで実行する場合に役立ちます。@@ ROWCOUNTは、更新された行数を示します。

ここに定義があります


0

これは私のために働きました:

DECLARE @SQL NVARCHAR(4000)

DECLARE @tbl Table (
    Id int,
    Account varchar(50),
    Amount int
) 

-- Lots of code to Create my dynamic sql statement

insert into @tbl EXEC sp_executesql @SQL

select * from @tbl
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.