mysql - 基于 LAST_INSERT_ID 的条件插入

标签 mysql

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

我希望能够有条件地插入一行,前提是之前的插入尝试实际上插入了一行 (ROW_COUNT > 0)。之前的插入可能是重复数据,所以我故意将其设置为 LAST_INSERT_ID为空,因此 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 表不存在重复数据问题,Events 表有一个复合唯一键,通过使用 INSERT IGNORE 防止重复数据。 :

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 可能有多行合法数据。

事件表下面可能有四个级别,具体取决于它是什么类型的数据,如果事件数据因为它是重复数据而没有被插入,那么也不会写入子数据(因为它会重复为好吧)。

最佳答案

您的第二次尝试几乎是正确的。您必须使用 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 的语法是

IF(expr1,expr2,expr3)

If expr1 is TRUE (expr1 <> 0 and expr1 <> NULL) then IF() returns expr2; otherwise it returns expr3. IF() returns a numeric or string value, depending on the context in which it is used.

语句的条件执行只能在 stored routines 中进行. IF syntax存储的例程将允许类似的东西

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

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

关于mysql - 基于 LAST_INSERT_ID 的条件插入,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25921124/

相关文章:

MySQL查询所有列的最小和最大字符长度

PHP MySQL - 使用 $_POST 更新

MySQL,同时在内部和外部查询中进行分组

python - Worker进程运行时异常 "heroku local"

php - 如何在 PHP 中从我的数据库中获取不可重复日期的列表?

php - yii2 如何在数据库列的一个特定字段中插入多个逗号分隔值

WHILE 语句中的 MySQL 语法错误

php - 使用 PHP 将 CSV 文件上传到数据库时出错

mysql - 我应该如何创建一个数据库表,在其中保留多个事件名称及其时间戳,并使用设备 token 作为外键

php - Laravel + 存储过程 : Error Packets out of order