Trigger for View

1) Trigger for View
1@@@@ basic case
referecing PLSQL Design 5th

  It is the most easy case, you could insert value into a view
  without considering CONSTRAINT and JOIN
    TEST@ocm> !cat test.sql
    DROP TABLE t_emp
    /
    CREATE TABLE t_emp (salary NUMBER, name VARCHAR2(20) )
    /
    CREATE OR REPLACE VIEW v_emp_rw
    AS
      SELECT salary FROM t_emp
    /
    CREATE OR REPLACE VIEW v_emp_ro
    AS
      SELECT salary FROM t_emp
    WITH READ ONLY
    /
    BEGIN
      INSERT INTO v_emp_rw VALUES(777);
    END;
    /
    prompt
    prompt @@@ You could not insert value into a read only view
    BEGIN
      INSERT INTO v_emp_ro VALUES(444);
    END;
    /
    SELECT * FROM v_emp_rw
    /
    SELECT * FROM v_emp_ro
    /
    prompt
    prompt @@@ It is possiable to insert values to a common view
    prompt @@@ which is read write without with read only.
    SELECT * FROM t_emp
    /
   
   @@@Now, Test the rw and ro view.
    TEST@ocm> @test.sql
    Table dropped.
    Table created.
    View created.
    View created.
    PL/SQL procedure successfully completed.
   
    @@@ You could not insert value into a read only view
    BEGIN
    *
    ERROR at line 1:
    ORA-01733: virtual column not allowed here
    ORA-06512: at line 2
   
        SALARY
    ----------
           777
   
        SALARY
    ----------
           777
   
    @@@ It is possiable to insert values to a common view
    @@@ which is read write without with read only.
   
        SALARY NAME
    ---------- --------------------
           777



2@@@@ complex case
 it is a join of multiple tables
  @@@Example: could not modify more than one base table
            though a join view.
    TEST@ocm> !cat test.sql
    DROP TABLE ta
    /
    DROP TABLE tb
    /
    CREATE TABLE tb
        (  b_id   NUMBER CONSTRAINT pk_b_id PRIMARY KEY
         , salary NUMBER )
    /
    CREATE TABLE ta
        (  a_id NUMBER 
         , name VARCHAR2(20)
             , b_id NUMBER
         , CONSTRAINT fk_b_id FOREIGN KEY (b_id)
             REFERENCING tb(b_id)  )
    /
    CREATE OR REPLACE VIEW v_ab
    AS
      SELECT a.name , b.salary
        FROM ta a , tb b
       WHERE a.b_id = b.b_id;
    /
    BEGIN
      INSERT INTO v_ab VALUES('Steven','Jack');
    END;
    /
    SELECT * FROM v_ab
    /
    SELECT * FROM ta
    /
    SELECT * FROM tb
    /
    TEST@ocm> @test.sql
    Table dropped.
    Table dropped.
    Table created.
    Table created.
    View created.
    View created.
    BEGIN
    *
    ERROR at line 1:
    ORA-01776: cannot modify more than one base table through a join view
    ORA-06512: at line 2
    no rows selected
    no rows selected
    no rows selected



3@@@@ Create trigger for complex view
INSTEAD OF Triggers
INSTEAD OF triggers control insert, update, merge, and delete operations on views,
not tables. They can be used to make nonupdateable views updateable and to override
the default behavior of views that are updateable.
To create (or replace) an INSTEAD OF trigger, use the syntax shown here:
    1 CREATE [OR REPLACE] TRIGGER trigger_name
    2   INTEAD OF operation
    3   ON view_name
    4   FOR EACH ROW
    5 BEGIN
    6    ...code goes here...
    7 END;
Note:
  This is where we see differences between INSTEAD OF triggers and
  other types of triggers. Because INSTEAD OF triggers are not really triggered
  by an event, I don not need to specify AFTER or BEFORE or provide an
  event name. What I do specify is the operation that the trigger is to fire in
  place of (or instead of). Stating INSTEAD OF followed by one of
  INSERT, UPDATE, MERGE, or DELETE accomplishes this.

    @@@Example: Create three table, and one table reference others, then create view
                from equal-join SELECT statement.
    HR@ocm> !cat tmp.sql
    DROP TABLE delivery
    /
    DROP TABLE driver
    /
    DROP TABLE area
    /
    CREATE TABLE area
        (  area_id    NUMBER
         , area_desc  VARCHAR2(30) CONSTRAINT u_area_desc UNIQUE
         , CONSTRAINT pk_area_id PRIMARY KEY (area_id, area_desc) )
    /
    CREATE TABLE driver
        (  driver_id   NUMBER
         , driver_name VARCHAR2(30) CONSTRAINT u_driver_name UNIQUE
         , CONSTRAINT pk_driver_id PRIMARY KEY (driver_id, driver_name) )
    /
    CREATE TABLE delivery
        (  delivery_id    NUMBER
         , delivery_start DATE
         , delivery_end   DATE
         , area_id        NUMBER
         , driver_id      NUMBER
         , CONSTRAINT pk_delivery_id
                  PRIMARY KEY (delivery_id)
         , CONSTRAINT fk_area_id  
                  FOREIGN KEY (area_id) REFERENCING area(area_id) 
         , CONSTRAINT fk_driver_id
              FOREIGN KEY (driver_id) REFERENCING driver(driver_id)  )
    /
    DROP SEQUENCE delivery_id_seq
    /
    DROP SEQUENCE area_id_seq
    /
    DROP SEQUENCE driver_id_seq
    /
    CREATE SEQUENCE delivery_id_seq
    /
    CREATE SEQUENCE area_id_seq
    /
    CREATE SEQUENCE driver_id_seq
    /
    CREATE OR REPLACE VIEW delivery_info
    AS
    SELECT    d.delivery_id
        , d.delivery_start
        , d.delivery_end
        , a.area_desc
        , dr.driver_name
    FROM   delivery  d
         , area      a
         , driver    dr
    WHERE  a.area_id = d.area_id
      AND  dr.driver_id = d.driver_id
    /
   
    HR@ocm> @tmp.sql
    Table dropped.
    Table dropped.
    Table dropped.
    Table created.
    Table created.
    Table created.
    Sequence dropped.
    Sequence dropped.
    Sequence dropped.
    Sequence created.
    Sequence created.
    Sequence created.
    View created.
   
  Because my system relies heavily on this view for query functionality, why not make it
  available for insert, update, and delete as well? I cannot directly issue DML statements
  against the view; it is a join of multiple tables. How would the database know what to
  do with an INSERT ? In fact, I need to tell the database very explicitly what to do when
  an insert, update, or delete operation occurs against the delivery_info view; in other
  words, I need to tell it what to do instead of trying to insert, update, or delete. Thus, I
  will use INSTEAD OF triggers. Let us start with the INSERT trigger.


My INSERT trigger will perform four basic operations:
 ~ Ensure that the delivery_end value is NULL. All delivery completions must be done
   via an update.
 ~ Try to find the driver ID based on the name provided. If the name cannot be found,
   then assign a new ID and create a driver entry using the name and the new ID.
 ~ Try to find the area ID based on the name provided. If the name cannot be found,
   then assign a new ID and create an area entry using the name and the new ID.
 ~ Create an entry in the delivery table.

Bear in mind that this example is intended to demonstrate triggers????ot how to effectively
build a business system! After a while I will probably wind up with a multitude
of duplicate driver and area entries. However, using this view speeds things up by not
requiring drivers and areas to be predefined, and in the fast-paced world of pizza delivery,
time is money!

   @@@Example01_insert: FOR a complex view with join, use trigger to implement DML.
    HR@ocm> !cat tmp.sql
    CREATE OR REPLACE TRIGGER delivery_info_insert
      INSTEAD OF INSERT
      ON delivery_info
    DECLARE
            /* Assure both driver_id and driver_name are primary key
               area_id and area_desc are primary key, at lesast unique*/
        --cursor to get the driver ID by :new.driver_name
        CURSOR curs_get_driver_id ( cp_driver_name_in IN VARCHAR2 )
        IS
         SELECT driver_id
           FROM driver
          WHERE driver_name = cp_driver_name_in;
        l_driver_id  NUMBER;
   
        --cursor to get the area ID by :new.area_desc
        CURSOR curs_get_area_id ( cp_area_desc_in  IN VARCHAR2 )
        IS
         SELECT area_id
           FROM area
          WHERE area_desc = cp_area_desc_in;
        l_area_id  NUMBER;
    BEGIN
       /* busniess logic: Make sure the delivery_end value is NULL */
       IF :new.delivery_end IS NOT NULL THEN
          RAISE_APPLICATION_ERROR( -20000,
          'Delivery end date value must be NULL when delivery createed');
       END IF;
   
       /* program_logic: use sequence to pad driver(driver_id)  */
       OPEN  curs_get_driver_id (  UPPER(:new.driver_name) );
       FETCH curs_get_driver_id INTO l_driver_id;
       IF curs_get_driver_id%NOTFOUND THEN
          SELECT driver_id_seq.nextval
            INTO l_driver_id
            FROM DUAL;
          INSERT INTO driver ( driver_id, driver_name )
               VALUES ( l_driver_id, UPPER(:new.driver_name) );
       END IF;
       CLOSE curs_get_driver_id;
     
       /* program_logic: use sequence to pad area(area_id)*/
       OPEN  curs_get_area_id(  UPPER(:new.driver_name)  );
       FETCH curs_get_area_id INTO l_area_id;
       IF curs_get_area_id%NOTFOUND THEN
          SELECT area_id_seq.nextval
            INTO l_area_id
            FROM DUAL;
          INSERT INTO area ( area_id, area_desc )
               VALUES ( l_area_id, UPPER(:new.area_desc) );
       END IF;
       CLOSE curs_get_area_id;
   
       INSERT INTO delivery
        (  delivery_id
         , delivery_start
         , delivery_end
         , area_id
         , driver_id )
       VALUES
        (  delivery_id_seq.nextval
         , NVL(:NEW.delivery_start, SYSDATE)
         , NULL
         , l_area_id
         , l_driver_id );
   
    END;
    /
   
    HR@ocm> @tmp.sql
   
    Trigger created.
   
   @@@Now test the trigger:
    HR@ocm> !cat tmpx.sql
    BEGIN
       INSERT INTO delivery_info
        (  delivery_id
         , delivery_start
         , delivery_end
         , area_desc
         , driver_name )
       VALUES
        (NULL, NULL, NULL, 'LOCAL COLLEGE', 'BIG TED' );
    END;
    /
    HR@ocm> @tmpx.sql
    PL/SQL procedure successfully completed.
    HR@ocm> SELECT * FROM delivery;
    DELIVERY_ID DELIVERY_START    DELIVERY_END           AREA_ID    DRIVER_ID
    ----------- ------------------- ------------------- ---------- ----------
          2 2013-01-05 15:52:43                  2        2


   @@@Example01_update:
    HR@ocm> !cat tmp.sql
    CREATE OR REPLACE TRIGGER delivery_info_update
      INSTEAD OF UPDATE
      ON delivery_info
    DECLARE
       --cursor to get the delivery entry
       CURSOR curs_get_delivery ( cp_delivery_id_in  IN  NUMBER )
       IS
        SELECT delivery_end
          FROM delivery
         WHERE delivery_id = cp_delivery_id_in
       FOR UPDATE OF delivery_end NOWAIT; --with CURRENT OF
       l_delivery_end DATE;
    BEGIN
       OPEN  curs_get_delivery(:NEW.delivery_id);
       FETCH curs_get_delivery INTO l_delivery_end;
       IF l_delivery_end IS NOT NULL THEN
          RAISE_APPLICATION_ERROR( -20000,
             'The delivery end date has already been set.');
       ELSE
          UPDATE delivery
             SET delivery_end = :new.delivery_end
           WHERE CURRENT OF curs_get_delivery;
       END IF;
       CLOSE curs_get_delivery;
    END;
    /
    HR@ocm> @tmp.sql
    Trigger created.
   
    HR@ocm> !cat test.sql
    BEGIN
       UPDATE delivery_info
          SET delivery_end = SYSDATE + 1
        WHERE driver_name = 'BIG TED';
    END;
    /
   
    HR@ocm> @test.sql
    PL/SQL procedure successfully completed.
   
    HR@ocm> SELECT delivery_end FROM delivery_info;
    DELIVERY_END
    -------------------
    2013-01-06 16:08:28
   
    HR@ocm> @test.sql
    BEGIN
    *
    ERROR at line 1:
    ORA-20000: The delivery end date has already been set.
    ORA-06512: at "HR.DELIVERY_INFO_UPDATE", line 15
    ORA-04088: error during execution of trigger 'HR.DELIVERY_INFO_UPDATE'
    ORA-06512: at line 2


    Note: without FOR UPDATE keyword would occur errors below:
    HR@ocm> show error;
    Errors for TRIGGER DELIVERY_INFO_UPDATE:
    LINE/COL ERROR
    -------- -----------------------------------------------------------------
    18/7     PL/SQL: SQL Statement ignored
    20/25     PL/SQL: ORA-00904: : invalid identifier
    20/25     PLS-00404: cursor 'CURS_GET_DELIVERY' must be declared with FOR
             UPDATE to use with CURRENT OF


   @@@Example01_delete: Obviously, it could not be deleted
    HR@ocm> !cat tmp.sql
    CREATE OR REPLACE TRIGGER delivery_info_delete
      INSTEAD OF DELETE
      ON delivery_info
    BEGIN
       -- business logic
       IF :new.delivery_end IS NOT NULL THEN
          RAISE_APPLICATION_ERROR(-20000,
           'Completed deliveries cannot be deleted');
       END IF;
       DELETE delivery
       WHERE delivery_id = :new.delivery_id;
    END;
    /
    HR@ocm> @tmp.sql
    Trigger created.
    HR@ocm> DELETE delivery_info WHERE delivery_id = 2;
    1 row deleted.
   
    HR@ocm> SELECT * FROM delivery;
    DELIVERY_ID DELIVERY_START    DELIVERY_END           AREA_ID    DRIVER_ID
    ----------- ------------------- ------------------- ---------- ----------
          2 2013-01-05 15:52:43 2013-01-06 16:08:28         2        2
   

你可能感兴趣的:(trigger,of,insead)