基于LAST_INSERT_ID的条件插入

时间:2014-09-18 19:45:23

标签: mysql

使用IF语句询问了一个几乎相同的问题here,但他没有得到可回答的答案,只是建议去here没有使用IF语句。我曾尝试使用后一个链接编写IF语句和条件语句,但我被卡住了(见下文)。

我希望只有在前一次插入尝试实际插入一行(ROW_COUNT > 0)时才能有条件地插入行。之前的插入可能是重复数据,因此我故意将其LAST_INSERT_ID设置为null,因此LAST_INSERT_ID不会发生后续子插入。 SQL脚本是由C#脚本创建的,因此LAST_INSERT_ID很可能没有指向您期望的位置。

以下是脚本生成代码的一个非常小的示例(最终数据库中有大约300万行):

SET @Vendors_Vendor_ID = (SELECT vendor_ID FROM VENDORS WHERE vendorName = 'PCA');
INSERT IGNORE INTO PCBID (PCBID, PCBDrawing, AssemblyDrawing, PONumber, Vendors_Vendor_ID) 
VALUES (11001, '10405', '41606', '091557.5', @Vendors_Vendor_ID);
SET @eventType_ID = (SELECT EVENTTYPE_ID FROM EVENTTYPES WHERE EVENTTYPE = 'Creation');
SET @USER = 'CTHOMAS';
INSERT IGNORE INTO EVENTS (PCBID, EVENTTYPE_ID, DATETIME, USER) 
VALUES (11001, @eventType_ID, '2009-06-15T13:15:27', @USER);
SET @EVENT_ID = IF(ROW_COUNT() > 0, LAST_INSERT_ID(), null);
-- THIS DOES NOT WORK
SELECT IF(@EVENT_ID != null, 
    INSERT INTO EVENTDETAILS (EVENT_ID, ITEMNAME, ITEMVALUE) 
    VALUES (@EVENT_ID, 'Notes', 'WO#89574'), 
    null);

-- THIS DOESN'T WORK EITHER
INSERT INTO EVENTDETAILS (EVENT_ID, ITEMNAME, ITEMVALUE)
SELECT @EVENT_ID, 'Notes', 'WO#89574'
WHERE @EVENT_ID != null;

PCBID 表对于重复数据不是问题,事件表有一个复合唯一键,可以使用INSERT IGNORE来阻止重复数据:< / p>

CREATE  TABLE IF NOT EXISTS `uniqueTest`.`events` (
`Event_ID` INT(11) NOT NULL AUTO_INCREMENT ,
`PCBID` INT(11) NOT NULL ,
`EventType_ID` INT(11) NOT NULL ,
`DateTime` DATETIME NOT NULL ,
`User` VARCHAR(45) NOT NULL ,
PRIMARY KEY (`Event_ID`) ,
UNIQUE KEY `PDU_Index` (`PCBID`, `DateTime`, `User`),

问题: 我需要能够基于之前插入 Events 表的插入来执行条件插入,如果它被忽略(因为它是重复数据),也不要插入任何子行。目前无法使任何EventDetail数据唯一,可能存在基于给定Event_ID的多行合法数据。

事件表下面有四个级别可能更深,具体取决于它是什么类型的数据,如果事件数据没有插入,因为它是重复数据,也没有写入子数据(因为它将被复制为孔)。

1 个答案:

答案 0 :(得分:1)

你的第二次尝试几乎是正确的。您必须使用IS NOT NULL检查NULL值。所以使用

INSERT INTO EVENTDETAILS (EVENT_ID, ITEMNAME, ITEMVALUE)
SELECT @EVENT_ID, 'Notes', 'WO#89574' FROM DUAL
WHERE @EVENT_ID IS NOT NULL;  -- instead of != 

INSERT INTO EVENTDETAILS (EVENT_ID, ITEMNAME, ITEMVALUE)
SELECT t.* FROM (
    SELECT @EVENT_ID, 'Notes', 'WO#89574'
) t
WHERE @EVENT_ID IS NOT NULL;  -- instead of != 

第一个无法运作:

-- THIS DOES NOT WORK
SELECT IF(@EVENT_ID != null, 
    INSERT INTO EVENTDETAILS (EVENT_ID, ITEMNAME, ITEMVALUE) ...

因为IF的语法是

  

如果(表达式1,表达式2,表达式3)

     

如果expr1为TRUE(expr1&lt;&gt; 0且expr1&lt;&gt; NULL),则IF()返回expr2;否则返回expr3。 IF()返回一个数字或字符串值,具体取决于使用它的上下文。

只能在stored routines中有条件地执行语句。存储例程的IF syntax将允许类似

的内容
IF @EVENT_ID IS NOT NULL THEN
    INSERT INTO EVENTDETAILS (EVENT_ID, ITEMNAME, ITEMVALUE) ...
END IF

您必须区分这两种语法版本。