启用和禁用另一个触发器内的触发器

时间:2021-02-25 10:04:36

标签: sql oracle oracle11g triggers

我有一张桌子位置

CREATE TABLE Location (
        idL     INTEGER,
        City    VARCHAR2(15) NOT NULL,
        Street  VARCHAR2(35) NOT NULL,
        Nation  CHAR(6) NOT NULL,
            
        CONSTRAINT PK_idL PRIMARY KEY(idL)
);

还有一张桌子

CREATE TABLE Person(
        p_Name  VARCHAR2(20) NOT NULL,
        p_Surname VARCHAR2(20) NOT NULL,
        idP     INTEGER,
        b_Date  DATE NOT NULL,
        id_PL    INTEGER,
 
        CONSTRAINT PK_idP PRIMARY KEY(idP),
        CONSTRAINT FK_idPL FOREIGN KEY(id_PL) REFERENCES Location(idL)
);

我“自动”计算主键如下:

CREATE SEQUENCE seq_loc_pk
start with 1
increment by 1;

CREATE OR REPLACE TRIGGER auto_pk_loc
BEFORE INSERT ON Location
FOR EACH ROW
BEGIN
    :new.idL := seq_loc_pk.nextval;
END;
/

现在我想为一个新人(当然是在我创建了正确的视图之后)插入 residence ,而不是这样的触发器:

CREATE OR REPLACE TRIGGER newperson
INSTEAD OF INSERT ON Residence
FOR EACH ROW
DECLARE
    nl Loc.idL%TYPE;
BEGIN
    ALTER TRIGGER auto_pk_loc DISABLE; -- Error
    nl := seq_loc_pk.nextval;
    :NEW.idL := nl;
    INSERT INTO Location VALUES(:NEW.City,:NEW.Street,:NEW.Nation);
    INSERT INTO Patient VALUES(:NEW.P_Name,:NEW.P_Surname,:NEW.B_Date,,nl);
    ALTER TRIGGER auto_pk_loc ENABLE;
END;
/

我考虑过禁用和启用触发器 auto_pk_loc 以便它不会无缘无故地创建额外的值,但我认为这不是正确的方法吗?它是什么?感谢回答的人。

1 个答案:

答案 0 :(得分:0)

您可以将它放在 execute immedaite 中:

BEGIN
    execute immedidate 'ALTER TRIGGER auto_pk_loc DISABLE'; 
    nl := seq_loc_pk.nextval;
    :NEW.idL := nl;
    INSERT INTO Location VALUES(:NEW.City,:NEW.Street,:NEW.Nation);
    INSERT INTO Patient VALUES(:NEW.P_Name,:NEW.P_Surname,:NEW.B_Date,,nl);
    execute immedidate 'ALTER TRIGGER auto_pk_loc ENABLE';
END;
/

但这会给你带来各种各样的问题; DDL 提交,因此您必须使其成为自主事务,并且会遇到并发问题。最好避免这种情况。

更好的方法是使用 returning 子句获取您刚刚插入的值:

BEGIN
    INSERT INTO Location VALUES(:NEW.City,:NEW.Street,:NEW.Nation)
      returning idl into nl;

    INSERT INTO Patient VALUES(:NEW.P_Name,:NEW.P_Surname,:NEW.B_Date,nl);
END;
/

尽管正如@astentx 所指出的,您可能希望使用 merge 来避免出现重复的位置。这不支持 returing,因此您必须使用 insert+update 的某种组合。

最后 - 假设您使用 12c 或更高版本 - 最好使用身份列或序列默认值通过触发器自动生成位置 ID。