Compound triggers v17

Defining a compound trigger on a table

This example shows a compound trigger that records a change to the employee salary by defining a compound trigger named hr_trigger on the emp table.

  1. Create a table named emp:

    CREATE TABLE emp(EMPNO INT, ENAME TEXT, SAL INT, DEPTNO INT);
    CREATE TABLE
  2. Create a compound trigger named hr_trigger. The trigger uses each of the four timing points to modify the salary with an INSERT, UPDATE, or DELETE statement. In the global declaration section, the initial salary is declared as 10,000.

    CREATE OR REPLACE TRIGGER hr_trigger
      FOR INSERT OR UPDATE OR DELETE ON emp
        COMPOUND TRIGGER
      -- Global declaration.
      var_sal NUMBER := 10000;
    
      BEFORE STATEMENT IS
      BEGIN
        var_sal := var_sal + 1000;
        DBMS_OUTPUT.PUT_LINE('Before Statement: ' || var_sal);
      END BEFORE STATEMENT;
    
      BEFORE EACH ROW IS
      BEGIN
        var_sal := var_sal + 1000;
        DBMS_OUTPUT.PUT_LINE('Before Each Row: ' || var_sal);
      END BEFORE EACH ROW;
    
      AFTER EACH ROW IS
      BEGIN
        var_sal := var_sal + 1000;
        DBMS_OUTPUT.PUT_LINE('After Each Row: ' || var_sal);
      END AFTER EACH ROW;
    
      AFTER STATEMENT IS
      BEGIN
        var_sal := var_sal + 1000;
        DBMS_OUTPUT.PUT_LINE('After Statement: ' || var_sal);
      END AFTER STATEMENT;
    
    END hr_trigger;
    
    Output: Trigger created.
  3. Insert the record into table emp:

    INSERT INTO emp (EMPNO, ENAME, SAL, DEPTNO) VALUES(1111,'SMITH', 10000, 20);

    The INSERT statement produces the following output:

    Output
    Before Statement: 11000
    Before each row: 12000
    After each row: 13000
    After statement: 14000
    INSERT 0 1
  4. The UPDATE statement updates the employee salary record, setting the salary to 15000 for a specific employee number:

    UPDATE emp SET SAL = 15000 where EMPNO = 1111;

    The UPDATE statement produces the following output:

    Before Statement: 11000
    Before each row: 12000
    After each row: 13000
    After statement: 14000
    UPDATE 1
    
    SELECT * FROM emp;
    Output
     EMPNO | ENAME |  SAL  | DEPTNO
    -------+-------+-------+--------
      1111 | SMITH | 15000 |     20
    (1 row)

DELETE

The DELETE statement deletes the employee salary record:

DELETE from emp where EMPNO = 1111;

The DELETE statement produces the following output:

Before Statement: 11000
Before each row: 12000
After each row: 13000
After statement: 14000
DELETE 1

SELECT * FROM emp;
Output
 EMPNO | ENAME | SAL | DEPTNO
-------+-------+-----+--------
(0 rows)

TRUNCATE

The TRUNCATE statement removes all the records from the emp table:

CREATE OR REPLACE TRIGGER hr_trigger
  FOR TRUNCATE ON emp
    COMPOUND TRIGGER
  -- Global declaration.
  var_sal NUMBER := 10000;
  BEFORE STATEMENT IS
  BEGIN
    var_sal := var_sal + 1000;
    DBMS_OUTPUT.PUT_LINE('Before Statement: ' || var_sal);
  END BEFORE STATEMENT;

  AFTER STATEMENT IS
  BEGIN
    var_sal := var_sal + 1000;
    DBMS_OUTPUT.PUT_LINE('After Statement: ' || var_sal);
  END AFTER STATEMENT;

END hr_trigger;

Output: Trigger created.

The TRUNCATE statement produces the following output:

TRUNCATE emp;
Output
Before Statement: 11000
After statement: 12000
TRUNCATE TABLE
Note

You can use the TRUNCATE statement only at a BEFORE STATEMENT or AFTER STATEMENT timing point.

Creating a compound trigger on a table with a WHEN condition

This example creates a compound trigger named hr_trigger on the emp table with a WHEN condition. The WHEN condition checks and prints the employee salary when an INSERT, UPDATE, or DELETE statement affects the emp table. The database evaluates the WHEN condition for a row-level trigger, and the trigger executes once per row if the WHEN condition evaluates to TRUE. The statement-level trigger executes regardless of the WHEN condition.

CREATE OR REPLACE TRIGGER hr_trigger
  FOR INSERT OR UPDATE OR DELETE ON emp
  REFERENCING NEW AS new OLD AS old
  WHEN (old.sal > 5000 OR new.sal < 8000)
    COMPOUND TRIGGER

  BEFORE STATEMENT IS
  BEGIN
    DBMS_OUTPUT.PUT_LINE('Before Statement');
  END BEFORE STATEMENT;

  BEFORE EACH ROW IS
  BEGIN
    DBMS_OUTPUT.PUT_LINE('Before Each Row: ' || :OLD.sal ||' ' || :NEW.sal);
  END BEFORE EACH ROW;

  AFTER EACH ROW IS
  BEGIN
    DBMS_OUTPUT.PUT_LINE('After Each Row: ' || :OLD.sal ||' ' || :NEW.sal);
  END AFTER EACH ROW;

  AFTER STATEMENT IS
  BEGIN
    DBMS_OUTPUT.PUT_LINE('After Statement');
  END AFTER STATEMENT;

END hr_trigger;

INSERT

Insert the record into table emp:

INSERT INTO emp(EMPNO, ENAME, SAL, DEPTNO) VALUES(1111, 'SMITH', 1600, 20);

The INSERT statement produces the following output:

Output
Before Statement
Before Each Row: 1600
After Each Row: 1600
After Statement
INSERT 0 1

UPDATE

The UPDATE statement updates the employee salary record, setting the salary to 7500:

UPDATE emp SET SAL = 7500 where EMPNO = 1111;

The UPDATE statement produces the following output:

Before Statement
Before Each Row: 1600 7500
After Each Row: 1600 7500
After Statement
UPDATE 1

SELECT * from emp;
Output
 empno | ename | sal  | deptno
-------+-------+------+--------
  1111 | SMITH | 7500 |     20
(1 row)

DELETE

The DELETE statement deletes the employee salary record:

DELETE from emp where EMPNO = 1111;

The DELETE statement produces the following output:

Before Statement
Before Each Row: 7500
After Each Row: 7500
After Statement
DELETE 1

SELECT * from emp;
Output
 empno | ename | sal | deptno
-------+-------+-----+--------
(0 rows)