次の方法で共有


FROM またはサブクエリを使用した UPDATE を実装する

適用対象: SQL Server Azure SQL データベース Azure SQL Managed Instance

Transact-SQL UPDATE ステートメント上のネイティブ コンパイル T-SQL モジュール内で次の構文要素はサポートされて "いません"。

  • FROM 句
  • サブクエリ

対照的に、上記の要素は、SELECT ステートメント上のネイティブ コンパイル モジュールではサポート "されています"。

FROM 句を含む UPDATE ステートメントは、多くの場合、テーブル値パラメーター (TVP) に基づいてテーブルの情報を更新するため、または AFTER トリガーでテーブルの列を更新するために使用されます。

TVP に基づく更新のシナリオについては、「 ネイティブ コンパイル ストアド プロシージャに MERGE 機能を実装する」をご覧ください。

次のサンプルは、トリガー内で実行される更新を示します。 表では、LastUpdated という名前の列が更新後の現在の日時に設定されています。 回避策では、次の項目を使用して個々の更新が実行されます。

  • IDENTITY 列を含むテーブル変数。
  • テーブル変数内で行を反復処理する WHILE ループ。

元の T-SQL UPDATE ステートメントを次に示します。

 UPDATE dbo.Table1  
     SET LastUpdated = SysDateTime()  
     FROM  
         dbo.Table1 t  
         JOIN Inserted i ON t.Id = i.Id;  

次のブロック内の T-SQL コードのサンプルは、良好なパフォーマンスを提供する回避策を示します。 回避策は、ネイティブ コンパイル トリガーで実装されます。 コードで注目すべき重要な点は次のとおりです。

  • dbo.Type1 という名前の型。これはメモリ最適化テーブル型です。
  • トリガーの WHILE ループ。
    • ループは、Inserted から一度に 1 行ずつ取得します。
DROP TABLE IF EXISTS dbo.Table1;  
GO

DROP TYPE IF EXISTS dbo.Type1;  
GO

-----------------------------
-- Table and table type.
-----------------------------
CREATE TABLE dbo.Table1 (
   Id INT NOT NULL PRIMARY KEY NONCLUSTERED,
   Column2 INT NOT NULL,
   LastUpdated DATETIME2 NOT NULL DEFAULT(SYSDATETIME())
)
WITH (MEMORY_OPTIMIZED = ON);
GO

CREATE TYPE dbo.Type1 AS TABLE (
   Id INT NOT NULL,
   RowID INT NOT NULL IDENTITY,
   INDEX ix_RowID HASH (RowID) WITH (BUCKET_COUNT = 1024)
)
WITH (MEMORY_OPTIMIZED = ON);
GO

----------------------------------------
-- Trigger that contains the workaround
-- for UPDATE with FROM.
----------------------------------------
CREATE TRIGGER dbo.tr_a_u_Table1 ON dbo.Table1
   WITH NATIVE_COMPILATION, SCHEMABINDING
   AFTER UPDATE
AS
BEGIN
   ATOMIC
   WITH (
       TRANSACTION ISOLATION LEVEL = SNAPSHOT,
       LANGUAGE = N'us_english'
   )

   DECLARE @tabvar1 dbo.Type1;

   INSERT @tabvar1 (Id)
   SELECT Id
   FROM Inserted;

   DECLARE @i INT = 1,
       @Id INT,
       @max INT = SCOPE_IDENTITY();

   ---- Loop as a workaround to simulate a cursor.
   ---- Iterate over the rows in the memory-optimized table  
   ----   variable and perform an update for each row.  
   WHILE @i <= @max
   BEGIN
       SELECT @Id = Id
       FROM @tabvar1
       WHERE RowID = @i;

       UPDATE dbo.Table1
       SET LastUpdated = SysDateTime()
       WHERE Id = @Id;

       SET @i += 1;
   END
END
GO

---------------------------------
-- Test to verify functionality.
---------------------------------
SET NOCOUNT ON;

INSERT dbo.Table1 (Id, Column2)
VALUES (1, 9), (2, 9), (3, 600);

SELECT N'BEFORE-Update' AS [BEFORE-Update], *
FROM dbo.Table1
ORDER BY Id;

WAITFOR DELAY '00:00:01';

UPDATE dbo.Table1
SET Column2 += 1
WHERE Column2 <= 99;

SELECT N'AFTER--Update' AS [AFTER--Update], *
FROM dbo.Table1
ORDER BY Id;
GO

結果セットは次のとおりです。

BEFORE-Update   Id   Column2   LastUpdated  
BEFORE-Update   1       9      2016-04-20 21:18:42.8394659  
BEFORE-Update   2       9      2016-04-20 21:18:42.8394659  
BEFORE-Update   3     600      2016-04-20 21:18:42.8394659  

AFTER--Update   Id   Column2   LastUpdated  
AFTER--Update   1      10      2016-04-20 21:18:43.8529692  
AFTER--Update   2      10      2016-04-20 21:18:43.8529692  
AFTER--Update   3     600      2016-04-20 21:18:42.8394659