เหตุใดคำสั่ง MERGE นี้จึงทำให้เซสชันถูกฆ่า


23

ฉันมีMERGEคำสั่งด้านล่างซึ่งออกกับฐานข้อมูล:

MERGE "MySchema"."Point" AS t
USING (
       SELECT "ObjectId", "PointName", z."Id" AS "LocationId", i."Id" AS "Region"
         FROM @p1 AS d
         JOIN "MySchema"."Region" AS i ON i."Name" = d."Region"
    LEFT JOIN "MySchema"."Location" AS z ON z."Name" = d."Location" AND z."Region" = i."Id"
       ) AS s
   ON s."ObjectId" = t."ObjectId"
 WHEN NOT MATCHED BY TARGET 
    THEN INSERT ("ObjectId", "Name", "LocationId", "Region") VALUES (s."ObjectId", s."PointName", s."LocationId", s."Region")
 WHEN MATCHED 
    THEN UPDATE 
     SET "Name" = s."PointName"
       , "LocationId" = s."LocationId"
       , "Region" = s."Region"
OUTPUT $action, inserted.*, deleted.*;

อย่างไรก็ตามสิ่งนี้ทำให้เซสชันถูกยกเลิกด้วยข้อผิดพลาดต่อไปนี้:

เกี่ยวกับข่าวสารเกี่ยวกับ 0, ระดับ 11, สถานะ 0, บรรทัด 67 ข้อผิดพลาดร้ายแรงที่เกิดขึ้นกับคำสั่งปัจจุบัน ควรยกเลิกผลลัพธ์หากมี

ข่าวสารเกี่ยวกับ 0, ระดับ 20, สถานะ 0, บรรทัด 67 เกิดข้อผิดพลาดที่รุนแรงในคำสั่งปัจจุบัน ควรยกเลิกผลลัพธ์หากมี

ฉันได้รวมสคริปต์ทดสอบสั้น ๆ เข้าด้วยกันซึ่งทำให้เกิดข้อผิดพลาด:

USE master;
GO
IF DB_ID('TEST') IS NOT NULL
DROP DATABASE "TEST";
GO
CREATE DATABASE "TEST";
GO
USE "TEST";
GO

SET NOCOUNT ON;

IF SCHEMA_ID('MySchema') IS NULL
EXECUTE('CREATE SCHEMA "MySchema"');
GO

IF OBJECT_ID('MySchema.Region', 'U') IS NULL
CREATE TABLE "MySchema"."Region" (
"Id" TINYINT IDENTITY NOT NULL CONSTRAINT "PK_MySchema_Region" PRIMARY KEY,
"Name" VARCHAR(8) NOT NULL CONSTRAINT "UK_MySchema_Region" UNIQUE
);
GO

INSERT [MySchema].[Region] ([Name]) 
VALUES (N'A'), (N'B'), (N'C'), (N'D'), (N'E'), ( N'F'), (N'G');

IF OBJECT_ID('MySchema.Location', 'U') IS NULL
CREATE TABLE "MySchema"."Location" (
"Id" SMALLINT IDENTITY NOT NULL CONSTRAINT "PK_MySchema_Location" PRIMARY KEY,
"Region" TINYINT NOT NULL CONSTRAINT "FK_MySchema_Location_Region" FOREIGN KEY REFERENCES "MySchema"."Region" ("Id"),
"Name" VARCHAR(128) NOT NULL,
CONSTRAINT "UK_MySchema_Location" UNIQUE ("Region", "Name") 
);
GO

IF OBJECT_ID('MySchema.Point', 'U') IS NULL
CREATE TABLE "MySchema"."Point" (
"ObjectId" BIGINT NOT NULL CONSTRAINT "PK_MySchema_Point" PRIMARY KEY,
"Name" VARCHAR(64) NOT NULL,
"LocationId" SMALLINT NULL CONSTRAINT "FK_MySchema_Point_Location" FOREIGN KEY REFERENCES "MySchema"."Location"("Id"),
"Region" TINYINT NOT NULL CONSTRAINT "FK_MySchema_Point_Region" FOREIGN KEY REFERENCES "MySchema"."Region" ("Id"),
CONSTRAINT "UK_MySchema_Point" UNIQUE ("Name", "Region", "LocationId")
);
GO

-- CONTAINS HISTORIC Point DATA
IF OBJECT_ID('MySchema.PointHistory', 'U') IS NULL
CREATE TABLE "MySchema"."PointHistory" (
"Id" BIGINT IDENTITY NOT NULL CONSTRAINT "PK_MySchema_PointHistory" PRIMARY KEY,
"ObjectId" BIGINT NOT NULL,
"Name" VARCHAR(64) NOT NULL,
"LocationId" SMALLINT NULL,
"Region" TINYINT NOT NULL
);
GO

CREATE TYPE "MySchema"."PointTable" AS TABLE (
"ObjectId"      BIGINT          NOT NULL PRIMARY KEY,
"PointName"     VARCHAR(64)     NOT NULL,
"Location"      VARCHAR(16)     NULL,
"Region"        VARCHAR(8)      NOT NULL,
UNIQUE ("PointName", "Region", "Location")
);
GO

DECLARE @p1 "MySchema"."PointTable";

insert into @p1 values(10001769996,N'ABCDEFGH',N'N/A',N'E')

MERGE "MySchema"."Point" AS t
USING (
       SELECT "ObjectId", "PointName", z."Id" AS "LocationId", i."Id" AS "Region"
         FROM @p1 AS d
         JOIN "MySchema"."Region" AS i ON i."Name" = d."Region"
    LEFT JOIN "MySchema"."Location" AS z ON z."Name" = d."Location" AND z."Region" = i."Id"
       ) AS s
   ON s."ObjectId" = t."ObjectId"
 WHEN NOT MATCHED BY TARGET 
    THEN INSERT ("ObjectId", "Name", "LocationId", "Region") VALUES (s."ObjectId", s."PointName", s."LocationId", s."Region")
 WHEN MATCHED 
    THEN UPDATE 
     SET "Name" = s."PointName"
       , "LocationId" = s."LocationId"
       , "Region" = s."Region"
OUTPUT $action, inserted.*, deleted.*;

หากฉันลบOUTPUTข้อผิดพลาดจะไม่เกิดขึ้น นอกจากนี้หากฉันลบการdeletedอ้างอิงดังนั้นข้อผิดพลาดจะไม่เกิดขึ้น ดังนั้นฉันดูเอกสาร MSDN สำหรับส่วนOUTPUTคำสั่งที่:

ลบไม่สามารถใช้กับคำสั่ง OUTPUT ในคำสั่ง INSERT

ซึ่งสมเหตุสมผลสำหรับฉันอย่างไรก็ตามประเด็นทั้งหมดMERGEคือคุณอาจไม่ทราบล่วงหน้า

นอกจากนี้สคริปต์ด้านล่างทำงานได้อย่างสมบูรณ์แบบโดยไม่คำนึงถึงการกระทำที่เกิดขึ้น:

USE tempdb;
GO
CREATE TABLE dbo.Target(EmployeeID int, EmployeeName varchar(10), 
     CONSTRAINT Target_PK PRIMARY KEY(EmployeeID));
CREATE TABLE dbo.Source(EmployeeID int, EmployeeName varchar(10), 
     CONSTRAINT Source_PK PRIMARY KEY(EmployeeID));
GO
INSERT dbo.Target(EmployeeID, EmployeeName) VALUES(100, 'Mary');
INSERT dbo.Target(EmployeeID, EmployeeName) VALUES(101, 'Sara');
INSERT dbo.Target(EmployeeID, EmployeeName) VALUES(102, 'Stefano');

GO
INSERT dbo.Source(EmployeeID, EmployeeName) Values(103, 'Bob');
INSERT dbo.Source(EmployeeID, EmployeeName) Values(104, 'Steve');
GO
-- MERGE statement with the join conditions specified correctly.
USE tempdb;
GO
BEGIN TRAN;
MERGE Target AS T
USING Source AS S
ON (T.EmployeeID = S.EmployeeID) 
WHEN NOT MATCHED BY TARGET AND S.EmployeeName LIKE 'S%' 
    THEN INSERT(EmployeeID, EmployeeName) VALUES(S.EmployeeID, S.EmployeeName)
WHEN MATCHED 
    THEN UPDATE SET T.EmployeeName = S.EmployeeName
WHEN NOT MATCHED BY SOURCE AND T.EmployeeName LIKE 'S%'
    THEN DELETE 
OUTPUT $action, inserted.*, deleted.*;
ROLLBACK TRAN;
GO 

นอกจากนี้ผมมีข้อสงสัยอื่น ๆ ที่ใช้OUTPUTในแบบเดียวกับที่มีการขว้างปาข้อผิดพลาดและพวกเขาทำงานสมบูรณ์ดี - MERGEแตกต่างระหว่างพวกเขาเป็นตารางที่มีส่วนร่วมใน

นี่เป็นสาเหตุสำคัญของการผลิตสำหรับเรา ฉันทำซ้ำข้อผิดพลาดนี้ใน SQL2014 และ SQL2016 ทั้ง VM และ Physical พร้อม RAM 128GB, 12 x 2.2GHZ Cores, Windows Server 2012 R2

แผนการดำเนินการโดยประมาณที่สร้างจากแบบสอบถามสามารถดูได้ที่นี่:

แผนการดำเนินการโดยประมาณ


1
แบบสอบถามสามารถสร้างแผนโดยประมาณได้หรือไม่? (นอกจากนี้สิ่งนี้จะไม่ทำให้ผู้คนจำนวนมากตกใจ แต่ฉันขอแนะนำวิธีการที่เก่าแก่ที่สุดต่อไป - คุณMERGEไม่มีHOLDLOCKสำหรับหนึ่งดังนั้นมันจึงไม่รอดพ้นจากสภาพการแข่งขันและยังมีข้อบกพร่องอื่น ๆ ที่ต้องพิจารณา หลังจากที่คุณแก้ไข - หรือรายงาน - สิ่งใดก็ตามที่ทำให้เกิดปัญหานี้)
แอรอนเบอร์ทรานด์ด์

1
มันให้กองการถ่ายโอนข้อมูลที่มีการละเมิดการเข้าถึง เท่าที่ฉันจะเห็นเมื่อคลี่คลายสแตกที่นี่i.stack.imgur.com/f9aWa.pngคุณควรยกระดับนี้ด้วย Microsoft PSS หากนี่เป็นสาเหตุของปัญหาที่สำคัญสำหรับคุณ โดยเฉพาะดูเหมือนว่าจะdeleted.ObjectIdเป็นสาเหตุของปัญหา OUTPUT $action, inserted.*, deleted.Name, deleted.LocationId, deleted.Regionทำงานได้ดี
Martin Smith

1
เห็นด้วยกับมาร์ติน ในขณะเดียวกันดูว่าคุณสามารถหลีกเลี่ยงปัญหานี้ได้โดยไม่ได้ใช้MySchema.PointTableประเภทและเพียงแค่ใช้เปลือยกายVALUES()ประโยคหรือตาราง #temp USINGหรือตัวแปรตารางภายใน อาจช่วยแยกปัจจัยที่สนับสนุน
Aaron Bertrand

ขอบคุณสำหรับความช่วยเหลือของคุณฉันลองใช้ตารางชั่วคราวและเกิดข้อผิดพลาดเดียวกัน ฉันจะเพิ่มมันด้วยการสนับสนุนผลิตภัณฑ์ - ในขณะเดียวกันฉันเขียนแบบสอบถามใหม่เพื่อไม่ใช้การผสานเพื่อให้เราสามารถทำงานต่อเนื่องได้
Mr.Brownstone

คำตอบ:


20

นี่เป็นข้อผิดพลาด

มันเกี่ยวข้องกับMERGE- การเพิ่มประสิทธิภาพการเติมช่องโหว่ที่เฉพาะเจาะจงที่ใช้เพื่อหลีกเลี่ยงการป้องกันฮาโลวีนอย่างชัดเจนและเพื่อกำจัดการเข้าร่วมและวิธีการโต้ตอบเหล่านี้กับคุณลักษณะแผนปรับปรุงอื่น ๆ

มีรายละเอียดเกี่ยวกับการเพิ่มประสิทธิภาพผู้ที่อยู่ในบทความของฉันมีปัญหาฮาโลวีน - ส่วนที่ 3

ของรางวัลคือการแทรกตามด้วยการผสานบนโต๊ะเดียวกัน :

ส่วนย่อยของแผน

วิธีการแก้ปัญหา

มีหลายวิธีในการเอาชนะการเพิ่มประสิทธิภาพนี้และหลีกเลี่ยงข้อผิดพลาด

  1. ใช้แฟล็กการติดตามที่ไม่มีเอกสารเพื่อบังคับใช้ Halloween Protection อย่างชัดเจน:

    OPTION (QUERYTRACEON 8692);
  2. เปลี่ยนONข้อเป็น:

    ON s."ObjectId" = t."ObjectId" + 0
  3. เปลี่ยนประเภทตารางPointTableเพื่อแทนที่คีย์หลักด้วย:

    ObjectID bigint NULL UNIQUE CLUSTERED CHECK (ObjectId IS NOT NULL)

    CHECKส่วนข้อ จำกัด ที่เป็นตัวเลือกรวมถึงเพื่อรักษาเดิม null ปฏิเสธทรัพย์สินของคีย์หลัก

การประมวลผลแบบสอบถามแบบง่าย '(การตรวจสอบคีย์ต่างประเทศการบำรุงรักษาดัชนีที่ไม่ซ้ำกันและคอลัมน์ผลลัพธ์) นั้นซับซ้อนพอที่จะเริ่มต้นด้วย การใช้MERGEเพิ่มเลเยอร์เพิ่มเติมหลาย ๆ รวมเข้ากับการเพิ่มประสิทธิภาพเฉพาะที่กล่าวถึงข้างต้นและคุณมีวิธีที่ดีในการพบข้อบกพร่องกรณีขอบเช่นนี้

หนึ่งที่เพิ่มเติมเพื่อเพิ่มสายยาวของข้อบกพร่องMERGEที่ได้รับรายงานด้วย

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.