Triggers

Download as txt, pdf, or txt
Download as txt, pdf, or txt
You are on page 1of 23

Schedule: Timing Topic

60 minutes Lecture
60 minutes Practice
120 minutes Total
Copyright Oracle Corporation, 2001. All rights reserved.
Creating Database Triggers
Oracle9i: Program with PL/SQL 16-2
16-2 Copyright Oracle Corporation, 2001. All rights reserved.
Objectives
After completing this lesson, you should be able to
do the following:
Describe different types of triggers
Describe database triggers and their use
Create database triggers
Describe database trigger firing rules
Remove database triggers
Lesson Aim
In this lesson, you learn how to create and use database triggers.
Oracle9i: Program with PL/SQL 16-3
16-3 Copyright Oracle Corporation, 2001. All rights reserved.
Types of Triggers
A trigger:
Is a PL/SQL block or a PL/SQL procedure
associated with a table, view, schema, or the
database
Executes implicitly whenever a particular event
takes place
Can be either:
Application trigger: Fires whenever an event occurs
with a particular application
Database trigger: Fires whenever a data event (such
as DML) or system event (such as logon or
shutdown) occurs on a schema or database
Types of Triggers
Application triggers execute implicitly whenever a particular data manipulation
language (DML) event
occurs within an application. An example of an application that uses triggers ex
tensively is one developed
with Oracle Forms Developer.
Database triggers execute implicitly when a data event such as DML on a table (a
n INSERT, UPDATE, or
DELETE triggering statement), an INSTEAD OF trigger on a view, or data definitio
n language (DDL)
statements such as CREATE and ALTER are issued, no matter which user is connecte
d or which application is
used. Database triggers also execute implicitly when some user actions or databa
se system actions occur, for
example, when a user logs on, or the DBA shut downs the database.
Note: Database triggers can be defined on tables and on views. If a DML operatio
n is issued on a view, the
INSTEAD OF trigger defines what actions take place. If these actions include DML
operations on tables, then
any triggers on the base tables are fired.
Database triggers can be system triggers on a database or a schema. With a datab
ase, triggers fire for each
event for all users; with a schema, triggers fire for each event for that specif
ic user.
This course covers creating database triggers. Creating database triggers based
on system events is discussed
in the lesson More Trigger Concepts.
Instructor Note
Database administration type triggers have been introduced in Oracle8i. Some of
the triggers that can be
created include user actions of logging on and off, and database shutdown and st
artup.
These types of triggers are covered in a later lesson.
Oracle9i: Program with PL/SQL 16-4
16-4 Copyright Oracle Corporation, 2001. All rights reserved.
Guidelines for Designing Triggers
Design triggers to:
Perform related actions
Centralize global operations
Do not design triggers:
Where functionality is already built into the Oracle
server
That duplicate other triggers
Create stored procedures and invoke them in a
trigger, if the PL/SQL code is very lengthy.
The excessive use of triggers can result in
complex interdependencies, which may be difficult
to maintain in large applications.
Guidelines for Designing Triggers
Use triggers to guarantee that when a specific operation is performed, related a
ctions are performed.
Use database triggers only for centralized, global operations that should be fir
ed for the triggering
statement, regardless of which user or application issues the statement.
Do not define triggers to duplicate or replace the functionality already built i
nto the Oracle database. For
example do not define triggers to implement integrity rules that can be done by
using declarative
constraints. An easy way to remember the design order for a business rule is to:
Use built-in constraints in the Oracle server such as, primary key, foreign key
and so on
Develop a database trigger or develop an application such as a servlet or Enterp
rise JavaBean
(EJB) on your middle tier
Use a presentation interface such as Oracle Forms, dynamic HTML, Java ServerPage
s (JSP) and
so on, if you cannot develop your business rule as mentioned above, which might
be a presentation
rule.
The excessive use of triggers can result in complex interdependencies, which may
be difficult to maintain
in large applications. Only use triggers when necessary, and beware of recursive
and cascading effects.
If the logic for the trigger is very lengthy, create stored procedures with the
logic and invoke them in the
trigger body.
Note that database triggers fire for every user each time the event occurs on wh
ich the trigger is created.
Oracle9i: Program with PL/SQL 16-5
16-5 Copyright Oracle Corporation, 2001. All rights reserved.
Application
INSERT INTO EMPLOYEES
. . .;
EMPLOYEES table CHECK_SAL trigger
Database Trigger: Example
Example of a Database Trigger
In this example, the database trigger CHECK_SAL checks salary values whenever an
y application tries to
insert a row into the EMPLOYEES table. Values that are out of range according to
the job category can be
rejected, or can be allowed and recorded in an audit table.
Instructor Note (for page 16-4)
Recursive trigger: This is a trigger that contains a DML operation changing the
very same table.
Cascading trigger: The action of one trigger cascades to another trigger, causin
g this second trigger to
fire. The Oracle server allows up to 32 triggers to cascade at any one time. How
ever, the number of
cascading triggers can be limited by changing the value of the OPEN_CURSORS data
base initialization
parameter, which is set to 50 by default.
Oracle9i: Program with PL/SQL 16-6
16-6 Copyright Oracle Corporation, 2001. All rights reserved.
Creating DML Triggers
A triggering statement contains:
Trigger timing
For table: BEFORE, AFTER
For view: INSTEAD OF
Triggering event: INSERT, UPDATE, or DELETE
Table name: On table, view
Trigger type: Row or statement
WHEN clause: Restricting condition
Trigger body: PL/SQL block
Database Trigger
Before coding the trigger body, decide on the values of the components of the tr
igger: the trigger timing, the
triggering event, and the trigger type.
If multiple triggers are defined for a table, be aware that the order in which m
ultiple triggers of the same type
fire is arbitrary. To ensure that triggers of the same type are fired in a parti
cular order, consolidate the triggers
into one trigger that calls separate procedures in the desired order.
Part Description Possible Values
Trigger timing When the trigger fires in relation to the
triggering event
BEFORE
AFTER
INSTEAD OF
Triggering event Which data manipulation operation on the
table or view causes the trigger to fire
INSERT
UPDATE
DELETE
Trigger type How many times the trigger body
executes
Statement
Row
Trigger body What action the trigger performs Complete PL/SQL block
Oracle9i: Program with PL/SQL 16-7
16-7 Copyright Oracle Corporation, 2001. All rights reserved.
DML Trigger Components
Trigger timing: When should the trigger fire?
BEFORE: Execute the trigger body before the
triggering DML event on a table.
AFTER: Execute the trigger body after the
triggering DML event on a table.
INSTEAD OF: Execute the trigger body instead of
the triggering statement. This is used for views
that are not otherwise modifiable.
BEFORE Triggers
This type of trigger is frequently used in the following situations:
To determine whether that triggering statement should be allowed to complete. (T
his situation enables
you to eliminate unnecessary processing of the triggering statement and its even
tual rollback in cases
where an exception is raised in the triggering action.)
To derive column values before completing a triggering INSERT or UPDATE statemen
t.
To initialize global variables or flags, and to validate complex business rules.
AFTER Triggers
This type of trigger is frequently used in the following situations:
To complete the triggering statement before executing the triggering action.
To perform different actions on the same triggering statement if a BEFORE trigge
r is already present.
INSTEAD OF Triggers
This type of trigger is used to provide a transparent way of modifying views tha
t cannot be modified directly
through SQL DML statements because the view is not inherently modifiable.
You can write INSERT, UPDATE, and DELETE statements against the view. The INSTEA
D OF trigger
works invisibly in the background performing the action coded in the trigger bod
y directly on the underlying
tables.
Oracle9i: Program with PL/SQL 16-8
16-8 Copyright Oracle Corporation, 2001. All rights reserved.
DML Trigger Components
Triggering user event: Which DML statement causes
the trigger to execute? You can use any of the
following:
INSERT
UPDATE
DELETE
The Triggering Event
The triggering event or statement can be an INSERT, UPDATE, or DELETE statement
on a table.
When the triggering event is an UPDATE statement, you can include a column list
to identify which
columns must be changed to fire the trigger. You cannot specify a column list fo
r an INSERT or for a
DELETE statement, because they always affect entire rows.
. . . UPDATE OF salary . . .
The triggering event can contain one, two, or all three of these DML operations.
. . . INSERT or UPDATE or DELETE
. . . INSERT or UPDATE OF job_id . . .
Oracle9i: Program with PL/SQL 16-9
16-9 Copyright Oracle Corporation, 2001. All rights reserved.
DML Trigger Components
Trigger type: Should the trigger body execute for each
row the statement affects or only once?
Statement: The trigger body executes once for the
triggering event. This is the default. A statement
trigger fires once, even if no rows are affected at all.
Row: The trigger body executes once for each row
affected by the triggering event. A row trigger is not
executed if the triggering event affects no rows.
Statement Triggers and Row Triggers
You can specify that the trigger will be executed once for every row affected by
the triggering statement
(such as a multiple row UPDATE) or once for the triggering statement, no matter
how many rows it affects.
Statement Trigger
A statement trigger is fired once on behalf of the triggering event, even if no
rows are affected at all.
Statement triggers are useful if the trigger action does not depend on the data
from rows that are affected or
on data provided by the triggering event itself: for example, a trigger that per
forms a complex security check
on the current user.
Row Trigger
A row trigger fires each time the table is affected by the triggering event. If
the triggering event affects no
rows, a row trigger is not executed.
Row triggers are useful if the trigger action depends on data of rows that are a
ffected or on data provided by
the triggering event itself.
Oracle9i: Program with PL/SQL 16-10
16-10 Copyright Oracle Corporation, 2001. All rights reserved.
DML Trigger Components
Trigger body: What action should the trigger perform?
The trigger body is a PL/SQL block or a call to a
procedure.
Trigger Body
The trigger action defines what needs to be done when the triggering event is is
sued. The PL/SQL block can
contain SQL and PL/SQL statements, and can define PL/SQL constructs such as vari
ables, cursors, exceptions,
and so on. You can also call a PL/SQL procedure or a Java procedure.
Additionally, row triggers use correlation names to access the old and new colum
n values of the row being
processed by the trigger.
Note: The size of a trigger cannot be more than 32 K.
Instructor Note
The Java procedure needs to be encapsulated in a PL/SQL wrapper. To call a Java
procedure, you use the CALL
statement. The syntax for the CALL statement to call a PL/SQL procedure is shown
in a later lesson.
Before creating any triggers, run the catproc.sql script while connected as SYS.
This script automatically
runs all of the scripts required for, or used within, the procedural extensions
to the Oracle server.
Oracle9i: Program with PL/SQL 16-11
16-11 Copyright Oracle Corporation, 2001. All rights reserved.
Firing Sequence
Triggering action
BEFORE statement
trigger
BEFORE row trigger
AFTER row trigger
AFTER statement trigger
DML statement
Use the following firing sequence for a trigger on a
table, when a single row is manipulated:
INSERT INTO departments (department_id,
department_name, location_id)
VALUES (400, 'CONSULTING', 2400);
Creating Row or Statement Triggers
Create a statement trigger or a row trigger based on the requirement that the tr
igger must fire once for each
row affected by the triggering statement, or just once for the triggering statem
ent, regardless of the number of
rows affected.
When the triggering data manipulation statement affects a single row, both the s
tatement trigger and the row
trigger fire exactly once.
Example
This SQL statement does not differentiate statement triggers from row triggers,
because exactly one row is
inserted into the table using this syntax.
Oracle9i: Program with PL/SQL 16-12
16-12 Copyright Oracle Corporation, 2001. All rights reserved.
UPDATE employees
SET salary = salary * 1.1
WHERE department_id = 30;
Firing Sequence
Use the following firing sequence for a trigger on a
table, when many rows are manipulated:
BEFORE statement trigger
BEFORE row trigger
.A.F.TER row trigger
BEFORE row trigger
.A.F.TER row trigger
AFTER statement trigger
Creating Row or Statement Triggers (continued)
When the triggering data manipulation statement affects many rows, the statement
trigger fires exactly once,
and the row trigger fires once for every row affected by the statement.
Example
The SQL statement in the slide above causes a row-level trigger to fire a number
of times equal to the number
of rows that satisfy the WHERE clause, that is, the number of employees reportin
g to department 30.
Oracle9i: Program with PL/SQL 16-13
16-13 Copyright Oracle Corporation, 2001. All rights reserved.
Syntax for Creating
DML Statement Triggers
CREATE [OR REPLACE] TRIGGER trigger_name
timing
event1 [OR event2 OR event3]
ON table_name
trigger_body
Note: Trigger names must be unique with respect to
other triggers in the same schema.
Syntax:
Syntax for Creating a Statement Trigger
Trigger names must be unique with respect to other triggers in the same schema.
Trigger names do not need
to be unique with respect to other schema objects, such as tables, views, and pr
ocedures.
Using column names along with the UPDATE clause in the trigger improves performa
nce, because the trigger
fires only when that particular column is updated and thus avoids unintended fir
ing when any other column is
updated.
trigger name Is the name of the trigger
timing Indicates the time when the trigger fires in relation to the
triggering event:
BEFORE
AFTER
event Identifies the data manipulation operation that causes the
trigger to fire:
INSERT
UPDATE [OF column]
DELETE
table/view_name Indicates the table associated with the trigger
trigger body Is the trigger body that defines the action performed by the
trigger, beginning with either DECLARE or BEGIN, ending
with END, or a call to a procedure
Oracle9i: Program with PL/SQL 16-14
16-14 Copyright Oracle Corporation, 2001. All rights reserved.
Creating DML Statement Triggers
CREATE OR REPLACE TRIGGER secure_emp
BEFORE INSERT ON employees
BEGIN
IF (TO_CHAR(SYSDATE,'DY') IN ('SAT','SUN')) OR
(TO_CHAR(SYSDATE,'HH24:MI')
NOT BETWEEN '08:00' AND '18:00')
THEN RAISE_APPLICATION_ERROR (-20500,'You may
insert into EMPLOYEES table only
during business hours.');
END IF;
END;
/
Example:
Creating DML Statement Triggers
You can create a BEFORE statement trigger in order to prevent the triggering ope
ration from succeeding if a
certain condition is violated.
For example, create a trigger to restrict inserts into the EMPLOYEES table to ce
rtain business hours, Monday
through Friday.
If a user attempts to insert a row into the EMPLOYEES table on Saturday, the use
r sees the message, the
trigger fails, and the triggering statement is rolled back. Remember that the
RAISE_APPLICATION_ERROR is a server-side built-in procedure that returns an erro
r to the user and
causes the PL/SQL block to fail.
When a database trigger fails, the triggering statement is automatically rolled
back by the Oracle
server.
Instructor Note
The time specified in the trigger in the slide is based on the database server s
ystem time. Hence, if you are in
a different time zone from the database server, your DML may not work even if yo
ur system clock is within
the time specified in the code.
Oracle9i: Program with PL/SQL 16-15
16-15 Copyright Oracle Corporation, 2001. All rights reserved.
Testing SECURE_EMP
INSERT INTO employees (employee_id, last_name,
first_name, email, hire_date,
job_id, salary, department_id)
VALUES (300, 'Smith', 'Rob', 'RSMITH', SYSDATE,
'IT_PROG', 4500, 60);
Example
Insert a row into the EMPLOYEES table during nonbusiness hours. When the date an
d time are out of the
business timings specified in the trigger, you get the error message as shown in
the slide.
Instructor Note
Note that the row might be inserted if you are in a different timezone from the
database server. The trigger
fires even if your system clock is within these business hours.
Oracle9i: Program with PL/SQL 16-16
16-16 Copyright Oracle Corporation, 2001. All rights reserved.
Using Conditional Predicates
CREATE OR REPLACE TRIGGER secure_emp
BEFORE INSERT OR UPDATE OR DELETE ON employees
BEGIN
IF (TO_CHAR (SYSDATE,'DY') IN ('SAT','SUN')) OR
(TO_CHAR (SYSDATE, 'HH24') NOT BETWEEN '08' AND '18')
THEN
IF DELETING THEN
RAISE_APPLICATION_ERROR (-20502,'You may delete from
EMPLOYEES table only during business hours.');
ELSIF INSERTING THEN
RAISE_APPLICATION_ERROR (-20500,'You may insert into
EMPLOYEES table only during business hours.');
ELSIF UPDATING ('SALARY') THEN
RAISE_APPLICATION_ERROR (-20503,'You may update
SALARY only during business hours.');
ELSE
RAISE_APPLICATION_ERROR (-20504,'You may update
EMPLOYEES table only during normal hours.');
END IF;
END IF;
END;
Combining Triggering Events
You can combine several triggering events into one by taking advantage of the sp
ecial conditional predicates
INSERTING, UPDATING, and DELETING within the trigger body.
Example
Create one trigger to restrict all data manipulation events on the EMPLOYEES tab
le to certain business hours,
Monday through Friday.
Oracle9i: Program with PL/SQL 16-17
16-17 Copyright Oracle Corporation, 2001. All rights reserved.
Creating a DML Row Trigger
CREATE [OR REPLACE] TRIGGER trigger_name
timing
event1 [OR event2 OR event3]
ON table_name
[REFERENCING OLD AS old | NEW AS new]
FOR EACH ROW
[WHEN (condition)]
trigger_body
Syntax:
Syntax for Creating a Row Trigger
trigger_name Is the name of the trigger
timing Indicates the time when the trigger fires in relation to the triggering e
vent:
BEFORE
AFTER
INSTEAD OF
event Identifies the data manipulation operation that causes the trigger to fire
:
INSERT
UPDATE [OF column]
DELETE
table_name Indicates the table associated with the trigger
REFERENCING Specifies correlation names for the old and new values of the curren
t row
(The default values are OLD and NEW)
FOR EACH ROW Designates that the trigger is a row trigger
WHEN
Specifies the trigger restriction; (This conditional predicate must be
enclosed in parenthesis and is evaluated for each row to determine whether
or not the trigger body is executed.)
trigger body Is the trigger body that defines the action performed by the trigge
r,
beginning with either DECLARE or BEGIN, ending with END, or a call to a
procedure
Oracle9i: Program with PL/SQL 16-18
16-18 Copyright Oracle Corporation, 2001. All rights reserved.
Creating DML Row Triggers
CREATE OR REPLACE TRIGGER restrict_salary
BEFORE INSERT OR UPDATE OF salary ON employees
FOR EACH ROW
BEGIN
IF NOT (:NEW.job_id IN ('AD_PRES', 'AD_VP'))
AND :NEW.salary > 15000
THEN
RAISE_APPLICATION_ERROR (-20202,'Employee
cannot earn this amount');
END IF;
END;
/
Creating a Row Trigger
You can create a BEFORE row trigger in order to prevent the triggering operation
from succeeding if a
certain condition is violated.
Create a trigger to allow only certain employees to be able to earn a salary of
more than 15,000.
If a user attempts to do this, the trigger raises an error.
UPDATE employees
SET salary = 15500
WHERE last_name = 'Russell';
Instructor Note
Alternatively, the value 15,000 in the program could be a value in a table that
can be retrieved into the
program, or a global variable.
Oracle9i: Program with PL/SQL 16-19
16-19 Copyright Oracle Corporation, 2001. All rights reserved.
Using OLD and NEW Qualifiers
CREATE OR REPLACE TRIGGER audit_emp_values
AFTER DELETE OR INSERT OR UPDATE ON employees
FOR EACH ROW
BEGIN
INSERT INTO audit_emp_table (user_name, timestamp,
id, old_last_name, new_last_name, old_title,
new_title, old_salary, new_salary)
VALUES (USER, SYSDATE, :OLD.employee_id,
:OLD.last_name, :NEW.last_name, :OLD.job_id,
:NEW.job_id, :OLD.salary, :NEW.salary );
END;
/
Using OLD and NEW Qualifiers
Within a ROW trigger, reference the value of a column before and after the data
change by prefixing it with
the OLD and NEW qualifier.
The OLD and NEW qualifiers are available only in ROW triggers.
Prefix these qualifiers with a colon (:) in every SQL and PL/SQL statement.
There is no colon (:) prefix if the qualifiers are referenced in the WHEN restri
cting condition.
Note: Row triggers can decrease the performance if you do a lot of updates on la
rger tables.
Instructor Note
If you run the above code and get a compilation error, check whether you have a
table called
AUDIT_EMP_TABLE; if not, run 16_addtabs.sql to create this table. Point out to t
he students that the
OLD and NEW are host variable created implicitly. The same rules apply to these
host variables in PL/SQL
bodies as in anonymous PL/SQL blocks.
Data Operation Old Value New Value
INSERT NULL Inserted value
UPDATE Value before update Value after update
DELETE Value before delete NULL
Oracle9i: Program with PL/SQL 16-20
16-20 Copyright Oracle Corporation, 2001. All rights reserved.
Using OLD and NEW Qualifiers:
Example Using Audit_Emp_Table
INSERT INTO employees
(employee_id, last_name, job_id, salary, ...)
VALUES (999, 'Temp emp', 'SA_REP', 1000, ...);
UPDATE employees
SET salary = 2000, last_name = 'Smith'
WHERE employee_id = 999;
SELECT user_name, timestamp, ... FROM audit_emp_table
Using OLD and NEW Qualifiers: Example Using AUDIT_EMP_TABLE
Create a trigger on the EMPLOYEES table to add rows to a user table, AUDIT_EMP_T
ABLE, logging a
user s activity against the EMPLOYEES table. The trigger records the values of sev
eral columns both before
and after the data changes by using the OLD and NEW qualifiers with the respecti
ve column name.
There is additional column COMMENTS in the AUDIT_EMP_TABLE that is not shown in
this slide.
Oracle9i: Program with PL/SQL 16-21
16-21 Copyright Oracle Corporation, 2001. All rights reserved.
Restricting a Row Trigger
CREATE OR REPLACE TRIGGER derive_commission_pct
BEFORE INSERT OR UPDATE OF salary ON employees
FOR EACH ROW
WHEN (NEW.job_id = 'SA_REP')
BEGIN
IF INSERTING
THEN :NEW.commission_pct := 0;
ELSIF :OLD.commission_pct IS NULL
THEN :NEW.commission_pct := 0;
ELSE
:NEW.commission_pct := :OLD.commission_pct + 0.05;
END IF;
END;
/
Example
To restrict the trigger action to those rows that satisfy a certain condition, p
rovide a WHEN clause.
Create a trigger on the EMPLOYEES table to calculate an employee s commission when
a row is added to the
EMPLOYEES table, or when an employee s salary is modified.
The NEW qualifier cannot be prefixed with a colon in the WHEN clause because the
WHEN clause is outside the
PL/SQL blocks.
Instructor Note
To assign values to columns using the NEW qualifier, create BEFORE ROW triggers.
If you attempt to create
an AFTER ROW trigger for the above code, you get a compilation error:
CREATE OR REPLACE TRIGGER derive_commission_pct
AFTER INSERT OR UPDATE OF salary ON employees
FOR EACH ROW WHEN (NEW.job_id = 'SA_REP')
BEGIN
IF INSERTING THEN :NEW.commission_pct := 0;
ELSIF :OLD.commission_pct IS NULL
THEN :NEW.commission_pct := 0;
ELSE :NEW.commission_pct :=
:OLD.commission_pct * (:NEW.salary/:OLD.salary);
END IF;
END;
/
CREATE OR REPLACE TRIGGER derive_commission_pct*
ERROR at line 1:
ORA-04084: cannot change NEW values for this trigger type
Oracle9i: Program with PL/SQL 16-22
16-22 Copyright Oracle Corporation, 2001. All rights reserved.
INSTEAD OF Triggers
Application
INSERT INTO my_view
. . .;
MY_VIEW
INSTEAD OF
Trigger
INSERT
TABLE1
UPDATE
TABLE2
INSTEAD OF Triggers
Use INSTEAD OF triggers to modify data in which the DML statement has been issue
d against an inherently
nonupdatable view. These triggers are called INSTEAD OF triggers because, unlike
other triggers, the Oracle
server fires the trigger instead of executing the triggering statement. This tri
gger is used to perform an
INSERT, UPDATE, or DELETE operation directly on the underlying tables.
You can write INSERT, UPDATE, or DELETE statements against a view, and the INSTE
AD OF trigger
works invisibly in the background to make the right actions take place.
Why Use INSTEAD OF Triggers?
A view cannot be modified by normal DML statements if the view query contains se
t operators, group
functions, clauses such as GROUP BY, CONNECT BY, START, the DISTINCT operator, o
r joins. For
example, if a view consists of more than one table, an insert to the view may en
tail an insertion into one table
and an update to another. So, you write an INSTEAD OF trigger that fires when yo
u write an insert against
the view. Instead of the original insertion, the trigger body executes, which re
sults in an insertion of data into
one table and an update to another table.
Note: If a view is inherently updateable and has INSTEAD OF triggers, the trigge
rs take precedence.
INSTEAD OF triggers are row triggers.
The CHECK option for views is not enforced when insertions or updates to the vie
w are performed by using
INSTEAD OF triggers. The INSTEAD OF trigger body must enforce the check.
Instructor Note
You can update a subset of join views.
Oracle9i: Program with PL/SQL 16-23
16-23 Copyright Oracle Corporation, 2001. All rights reserved.
Creating an INSTEAD OF Trigger
CREATE [OR REPLACE] TRIGGER trigger_name
INSTEAD OF
event1 [OR event2 OR event3]
ON view_name
[REFERENCING OLD AS old | NEW AS new]
[FOR EACH ROW]
trigger_body
Syntax:
Syntax for Creating an INSTEAD OF Trigger
Note: INSTEAD OF triggers can be written only for views. BEFORE and AFTER option
s are not valid.
trigger_name Is the name of the trigger.
INSTEAD OF Indicates that the trigger belongs to a view
event Identifies the data manipulation operation that causes the trigger
to fire:
INSERT
UPDATE [OF column]
DELETE
view_name Indicates the view associated with trigger
REFERENCING Specifies correlation names for the old and new values of the
current row (The defaults are OLD and NEW)
FOR EACH
ROW
Designates the trigger to be a row trigger; INSTEAD OF triggers
can only be row triggers: if this is omitted, the trigger is still
defined as a row trigger.
trigger body Is the trigger body that defines the action performed by the
trigger, beginning with either DECLARE or BEGIN, and ending
with END or a call to a procedure
Oracle9i: Program with PL/SQL 16-24
Creating an INSTEAD OF Trigger
Example:
The following example creates two new tables, NEW_EMPS and NEW_DEPTS, based on t
he EMPLOYEES
and DEPARTMENTS tables respectively. It also creates an EMP_DETAILS view from th
e EMPLOYEES and
DEPARTMENTS tables. The example also creates an INSTEAD OF trigger, NEW_EMP_DEPT
. When a row is
inserted into the EMP_DETAILS view, instead of inserting the row directly into t
he view, rows are added
into the NEW_EMPS and NEW_DEPTS tables, based on the data in the INSERT statemen
t. Similarly, when a
row is modified or deleted through the EMP_DETAILS view, corresponding rows in t
he NEW_EMPS and
NEW_DEPTS tables are affected.
CREATE TABLE new_emps AS
SELECT employee_id, last_name, salary, department_id,
email, job_id, hire_date
FROM employees;
CREATE TABLE new_depts AS
SELECT d.department_id, d.department_name, d.location_id,
sum(e.salary) tot_dept_sal
FROM employees e, departments d
WHERE e.department_id = d.department_id
GROUP BY d.department_id, d.department_name, d.location_id;
CREATE VIEW emp_details AS
SELECT e.employee_id, e.last_name, e.salary, e.department_id,
e.email, e.job_id, d.department_name, d.location_id
FROM employees e, departments d
WHERE e.department_id = d.department_id;
CREATE OR REPLACE TRIGGER new_emp_dept
INSTEAD OF INSERT OR UPDATE OR DELETE ON emp_details
FOR EACH ROW
BEGIN
IF INSERTING THEN
INSERT INTO new_emps
VALUES (:NEW.employee_id, :NEW.last_name, :NEW.salary,
:NEW.department_id, :NEW.email, :New.job_id, SYSDATE);
UPDATE new_depts
SET tot_dept_sal = tot_dept_sal + :NEW.salary
WHERE department_id = :NEW.department_id;
ELSIF DELETING THEN
DELETE FROM new_emps
WHERE employee_id = :OLD.employee_id;
UPDATE new_depts
SET tot_dept_sal = tot_dept_sal - :OLD.salary
WHERE department_id = :OLD.department_id;
Oracle9i: Program with PL/SQL 16-25
Creating an INSTEAD OF Trigger (continued)
Example:
ELSIF UPDATING ('salary')
THEN
UPDATE new_emps
SET salary = :NEW.salary
WHERE employee_id = :OLD.employee_id;
UPDATE new_depts
SET tot_dept_sal = tot_dept_sal + (:NEW.salary - :OLD.salary)
WHERE department_id = :OLD.department_id;
ELSIF UPDATING ('department_id')
THEN
UPDATE new_emps
SET department_id = :NEW.department_id
WHERE employee_id = :OLD.employee_id;
UPDATE new_depts
SET tot_dept_sal = tot_dept_sal - :OLD.salary
WHERE department_id = :OLD.department_id;
UPDATE new_depts
SET tot_dept_sal = tot_dept_sal + :NEW.salary
WHERE department_id = :NEW.department_id;
END IF;
END;
/
Note: This example is explained in the next page by using graphics.
Instructor Note
Demonstration: 16_inst_of.sql
Purpose: Creates the two new tables, a view and the trigger. The script shows th
e data in the view and the
tables. A direct INSERT into the view implicitly executes the trigger, which INS
ERTs into the NEW_EMPS
table and UPDATEs the NEW_DEPTS table.
The trigger is not complete. It is programmed to handle the INSERT and UPDATE of
the SALARY or
DEPARTMENT_ID, and a DELETE. Any other form of UPDATE is not programmed.
Oracle9i: Program with PL/SQL 16-26
16-26 Copyright Oracle Corporation, 2001. All rights reserved.
Creating an INSTEAD OF Trigger
INSERT INTO emp_details(employee_id, ... )
VALUES(9001,'ABBOTT',3000,10,'abbott.mail.com','HR_MAN');
INSERT into EMP_DETAILS that is based on EMPLOYEES and
DEPARTMENTS tables
INSTEAD OF INSERT
into EMP_DETAILS
1
Creating an INSTEAD OF Trigger
You can create an INSTEAD OF trigger in order to maintain the base tables on whi
ch a view is based.
Assume that an employee name will be inserted using the view EMP_DETAILS that is
created based on the
EMPLOYEES and DEPARTMENTS tables. Create a trigger that results in the appropria
te INSERT and
UPDATE to the base tables. The slide in the next page explains how an INSTEAD OF
TRIGGER behaves in
this situation.
Oracle9i: Program with PL/SQL 16-27
16-27 Copyright Oracle Corporation, 2001. All rights reserved.
Creating an INSTEAD OF Trigger
INSERT INTO emp_details(employee_id, ... )
VALUES(9001,'ABBOTT',3000,10,'abbott.mail.com','HR_MAN');
INSERT into EMP_DETAILS that is based on EMPLOYEES and
DEPARTMENTS tables
INSTEAD OF INSERT
into EMP_DETAILS
INSERT into
NEW_EMPS
UPDATE
NEW_DEPTS

1
2 3
Creating an INSTEAD OF Trigger
Because of the INSTEAD OF TRIGGER on the view EMP_DETAILS, instead of inserting
the new employee
record into the EMPLOYEES table:
A row is inserted into the NEW_EMPS table.
The TOTAL_DEPT_SAL column of the NEW_DEPTS table is updated. The salary value su
pplied for
the new employee is added to the existing total salary of the department to whic
h the new employee has
been assigned.
Oracle9i: Program with PL/SQL 16-28
16-28 Copyright Oracle Corporation, 2001. All rights reserved.
Differentiating Between Database Triggers
and Stored Procedures
Triggers
Defined with CREATE TRIGGER
Data dictionary contains source
code in USER_TRIGGERS
Implicitly invoked
COMMIT, SAVEPOINT, and
ROLLBACK are not allowed
Procedures
Defined with CREATE PROCEDURE
Data dictionary contains source code
in USER_SOURCE
Explicitly invoked
COMMIT, SAVEPOINT, and ROLLBACK
are allowed
Database Triggers and Stored Procedures
There are differences between database triggers and stored procedures:
Triggers are fully compiled when the CREATE TRIGGER command is issued and the P
code is stored in the
data dictionary.
If errors occur during the compilation of a trigger, the trigger is still create
d.
Database Trigger Stored Procedure
Invoked implicitly Invoked explicitly
COMMIT, ROLLBACK, and
SAVEPOINT statements are not allowed
within the trigger body. It is possible to
commit or rollback indirectly by calling a
procedure, but it is not recommended
because of side effects to transactions.
COMMIT, ROLLBACK, and SAVEPOINT
statements are permitted within the procedure
body.
Oracle9i: Program with PL/SQL 16-29
16-29 Copyright Oracle Corporation, 2001. All rights reserved.
Differentiating Between Database Triggers
and Form Builder Triggers
INSERT INTO EMPLOYEES
. . .;
EMPLOYEES table CHECK_SAL trigger
BEFORE
INSERT
row
Differences between a Database Trigger and a Form Builder Trigger
Database triggers are different from Form Builder triggers.
Database Trigger Form Builder Trigger
Executed by actions from any database tool
or application
Executed only within a particular Form Builder
application
Always triggered by a SQL DML, DDL, or a
certain database action
Can be triggered by navigating from field to field, by
pressing a key, or by many other actions
Is distinguished as either a statement or row
trigger
Is distinguished as a statement or row trigger
Upon failure, causes the triggering statement
to roll back
Upon failure, causes the cursor to freeze and may
cause the entire transaction to roll back
Fires independently of, and in addition to,
Form Builder triggers
Fires independently of, and in addition to, database
triggers
Executes under the security domain of the
author of the trigger
Executes under the security domain of the Form
Builder user
Oracle9i: Program with PL/SQL 16-30
16-30 Copyright Oracle Corporation, 2001. All rights reserved.
ALTER TRIGGER trigger_name DISABLE | ENABLE
Managing Triggers
Disable or reenable a database trigger:
ALTER TABLE table_name DISABLE | ENABLE ALL TRIGGERS
Disable or reenable all triggers for a table:
ALTER TRIGGER trigger_name COMPILE
Recompile a trigger for a table:
Trigger Modes: Enabled or Disabled
When a trigger is first created, it is enabled automatically.
The Oracle server checks integrity constraints for enabled triggers and guarante
es that triggers cannot
compromise them. In addition, the Oracle server provides read-consistent views f
or queries and
constraints, manages the dependencies, and provides a two-phase commit process i
f a trigger updates
remote tables in a distributed database.
Disable a specific trigger by using the ALTER TRIGGER syntax, or disable all tri
ggers on a table by
using the ALTER TABLE syntax.
Disable a trigger to improve performance or to avoid data integrity checks when
loading massive
amounts of data by using utilities such as SQL*Loader. You may also want to disa
ble the trigger when
it references a database object that is currently unavailable, owing to a failed
network connection, disk
crash, offline data file, or offline tablespace.
Compile a Trigger
Use the ALTER TRIGGER command to explicitly recompile a trigger that is invalid.
When you issue an ALTER TRIGGER statement with the COMPILE option, the trigger r
ecompiles,
regardless of whether it is valid or invalid.
Oracle9i: Program with PL/SQL 16-31
16-31 Copyright Oracle Corporation, 2001. All rights reserved.
DROP TRIGGER Syntax
To remove a trigger from the database, use the DROP
TRIGGER syntax:
DROP TRIGGER trigger_name;
DROP TRIGGER secure_emp;
Example:
Note: All triggers on a table are dropped when the
table is dropped.
Removing Triggers
When a trigger is no longer required, you can use a SQL statement in iSQL*Plus t
o drop it.
Oracle9i: Program with PL/SQL 16-32
16-32 Copyright Oracle Corporation, 2001. All rights reserved.
Trigger Test Cases
Test each triggering data operation, as well as
nontriggering data operations.
Test each case of the WHEN clause.
Cause the trigger to fire directly from a basic data
operation, as well as indirectly from a procedure.
Test the effect of the trigger upon other triggers.
Test the effect of other triggers upon the trigger.
Testing Triggers
Ensure that the trigger works properly by testing a number of cases separately.
Take advantage of the DBMS_OUTPUT procedures to debug triggers. You can also use
the Procedure
Builder debugging tool to debug triggers. Using Procedure Builder is discussed i
n Appendix F,
Creating Program Units by Using Procedure Builder.
Instructor Note
When writing new application code with procedures and functions, you should veri
fy the USER_TRIGGERS
data dictionary view for any existing triggers. The newly added procedures and f
unctions may fire these
triggers implicitly, which is not intended. Hence, test the procedures and funct
ions, and drop any triggers that
are not required.
Oracle9i: Program with PL/SQL 16-33
16-33 Copyright Oracle Corporation, 2001. All rights reserved.
Trigger Execution Model
and Constraint Checking
1. Execute all BEFORE STATEMENT triggers.
2. Loop for each row affected:
a. Execute all BEFORE ROW triggers.
b. Execute all AFTER ROW triggers.
3. Execute the DML statement and perform integrity
constraint checking.
4. Execute all AFTER STATEMENT triggers.
Trigger Execution Model
A single DML statement can potentially fire up to four types of triggers: BEFORE
and AFTER statement and
row triggers. A triggering event or a statement within the trigger can cause one
or more integrity constraints
to be checked. Triggers can also cause other triggers to fire (cascading trigger
s).
All actions and checks done as a result of a SQL statement must succeed. If an e
xception is raised within a
trigger and the exception is not explicitly handled, all actions performed becau
se of the original SQL
statement are rolled back. This includes actions performed by firing triggers. T
his guarantees that integrity
constraints can never be compromised by triggers.
When a trigger fires, the tables referenced in the trigger action may undergo ch
anges by other users'
transactions. In all cases, a read-consistent image is guaranteed for modified v
alues the trigger needs to read
(query) or write (update).
Instructor Note
The example in the following slide demonstrates the constraint verification proc
edure. You can demonstrate
this concept by using the file 16_trig_constr.sql.
Oracle9i: Program with PL/SQL 16-34
16-34 Copyright Oracle Corporation, 2001. All rights reserved.
Trigger Execution Model and Constraint
Checking: Example
CREATE OR REPLACE TRIGGER constr_emp_trig
AFTER UPDATE ON employees
FOR EACH ROW
BEGIN
INSERT INTO departments
VALUES (999, 'dept999', 140, 2400);
END;
/
UPDATE employees SET department_id = 999
WHERE employee_id = 170;
-- Successful after trigger is fired
UPDATE employees SET department_id = 999
WHERE employee_id = 170;
-- Integrity constraint violation error
Trigger Execution Model and Constraint Checking: Example
The example in the slide explains a situation in which the integrity constraint
can be taken care of by using a
trigger. Table EMPLOYEES has a foreign key constraint on the DEPARTMENT_ID colum
n of the
DEPARTMENTS table.
In the first SQL statement, the DEPARTMENT_ID of the employee with EMPLOYEE_ID 1
70 is modified to
999.
Because such a department does not exist in the DEPARTMENTS table, the statement
raises the exception -
2292 for the integrity constraint violation.
A trigger CONSTR_EMP_TRIG is created that inserts a new department 999 into the
DEPARTMENTS table.
When the UPDATE statement that modifies the department of employee 170 to 999 is
issued, the trigger fires.
Then, the foreign key constraint is checked. Because the trigger inserted the de
partment 999 into the
DEPARTMENTS table, the foreign key constraint check is successful and there is n
o exception.
This process works with Oracle8i and later releases. The example described in th
e slide produces a run-time
error in releases prior to Oracle8i.
Oracle9i: Program with PL/SQL 16-35
16-35 Copyright Oracle Corporation, 2001. All rights reserved.
VAR_PACK
package
AUDIT_EMP_TRIG
FOR EACH ROW
Increment variables
2
A Sample Demonstration for Triggers
Using Package Constructs
AUDIT_EMP_TAB
AFTER STATEMENT
Copy and then reset
variables
3
DML into
EMPLOYEES table
1
AUDIT_TABLE
4
A Sample Demonstration
The following pages of PL/SQL subprograms are an example of the interaction of t
riggers, packaged
procedures, functions, and global variables.
The sequence of events:
1. Issue an INSERT, UPDATE, or DELETE command that can manipulate one or many ro
ws.
2. AUDIT_EMP_TRIG, the AFTER ROW trigger, calls the packaged procedure to increm
ent the global
variables in the package VAR_PACK. Because this is a row trigger, the trigger fi
res once for each row
that you updated.
3. When the statement has finished, AUDIT_EMP_TAB, the AFTER STATEMENT trigger,
calls the
procedure AUDIT_EMP.
4. This procedure assigns the values of the global variables into local variable
s using the packaged
functions, updates the AUDIT_TABLE, and then resets the global variables.
Instructor Note
The slide presents a simplified version of the demonstration. First run the scri
pt 16_drop_trg.sql. The
following demonstration scripts contain a complete explanation about the demonst
ration as described in the
next few pages.
Demonstration: 16_comp_ex_spec.sql and 16_comp_ex_body.sql
Purpose: To create the VAR_PACK package, the AUDIT_EMP procedure, and the AUDIT_
EMP_TRIG
trigger. If you wish to complement the code by running an example, insert a row
into AUDIT_TABLE with
your user name and EMPLOYEES as the table name, and then write an INSERT, UPDATE
, or DELETE
command to see the triggers working.
Oracle9i: Program with PL/SQL 16-36
16-36 Copyright Oracle Corporation, 2001. All rights reserved.
After Row and After Statement Triggers
CREATE OR REPLACE TRIGGER audit_emp_trig
AFTER UPDATE or INSERT or DELETE on EMPLOYEES
FOR EACH ROW
BEGIN
IF DELETING THEN var_pack.set_g_del(1);
ELSIF INSERTING THEN var_pack.set_g_ins(1);
ELSIF UPDATING ('SALARY')
THEN var_pack.set_g_up_sal(1);
ELSE var_pack.set_g_upd(1);
END IF;
END audit_emp_trig;
/
CREATE OR REPLACE TRIGGER audit_emp_tab
AFTER UPDATE or INSERT or DELETE on employees
BEGIN
audit_emp;
END audit_emp_tab;
/
AFTER Row and AFTER Statement Triggers
The trigger AUDIT_EMP_TRIG is a row trigger that fires after every row manipulat
ed. This trigger invokes
the package procedures depending on the type of DML performed. For example, if t
he DML updates salary
of an employee, then the trigger invokes the procedure SET_G_UP_SAL. This packag
e procedure inturn
invokes the function G_UP_SAL. This function increments the package variable GV_
UP_SAL that keeps
account of the number of rows being changed due to update of the salary.
The trigger AUDIT_EMP_TAB will fire after the statement has finished. This trigg
er invokes the procedure
AUDIT_EMP, which is on the following pages. The AUDIT_EMP procedure updates the
AUDIT_TABLE
table. An entry is made into the AUDIT_TABLE table with the information such as
the user who performed
the DML, the table on which DML is performed, and the total number of such data
manipulations performed
so far on the table (indicated by the value of the corresponding column in the A
UDIT_TABLE table). At the
end, the AUDIT_EMP procedure resets the package variables to 0.
Instructor Note
This is not the only way to implement this business requirement, but it does sho
w the way packages, triggers,
procedures, and packaged variables can interact.
Oracle9i: Program with PL/SQL 16-37
16-37 Copyright Oracle Corporation, 2001. All rights reserved.
Demonstration: VAR_PACK Package
Specification
CREATE OR REPLACE PACKAGE var_pack
IS
-- these functions are used to return the
-- values of package variables
FUNCTION g_del RETURN NUMBER;
FUNCTION g_ins RETURN NUMBER;
FUNCTION g_upd RETURN NUMBER;
FUNCTION g_up_sal RETURN NUMBER;
-- these procedures are used to modify the
-- values of the package variables
PROCEDURE set_g_del (p_val IN NUMBER);
PROCEDURE set_g_ins (p_val IN NUMBER);
PROCEDURE set_g_upd (p_val IN NUMBER);
PROCEDURE set_g_up_sal (p_val IN NUMBER);
END var_pack;
/
var_pack.sql
Demonstration: VAR_PACK Package Body
var_pack_body.sql
CREATE OR REPLACE PACKAGE BODY var_pack IS
gv_del NUMBER := 0; gv_ins NUMBER := 0;
gv_upd NUMBER := 0; gv_up_sal NUMBER := 0;
FUNCTION g_del RETURN NUMBER IS
BEGIN
RETURN gv_del;
END;
FUNCTION g_ins RETURN NUMBER IS
BEGIN
RETURN gv_ins;
END;
FUNCTION g_upd RETURN NUMBER IS
BEGIN
RETURN gv_upd;
END;
FUNCTION g_up_sal RETURN NUMBER IS
BEGIN
RETURN gv_up_sal;
END;
(continued on the next page)
Oracle9i: Program with PL/SQL 16-38
VAR_PACK Package Body (continued)
PROCEDURE set_g_del (p_val IN NUMBER) IS
BEGIN
IF p_val = 0 THEN
gv_del := p_val;
ELSE gv_del := gv_del +1;
END IF;
END set_g_del;
PROCEDURE set_g_ins (p_val IN NUMBER) IS
BEGIN
IF p_val = 0 THEN
gv_ins := p_val;
ELSE gv_ins := gv_ins +1;
END IF;
END set_g_ins;
PROCEDURE set_g_upd (p_val IN NUMBER) IS
BEGIN
IF p_val = 0 THEN
gv_upd := p_val;
ELSE gv_upd := gv_upd +1;
END IF;
END set_g_upd;
PROCEDURE set_g_up_sal (p_val IN NUMBER) IS
BEGIN
IF p_val = 0 THEN
gv_up_sal := p_val;
ELSE gv_up_sal := gv_up_sal +1;
END IF;
END set_g_up_sal;
END var_pack;
/
Oracle9i: Program with PL/SQL 16-39
16-39 Copyright Oracle Corporation, 2001. All rights reserved.
CREATE OR REPLACE PROCEDURE audit_emp IS
v_del NUMBER := var_pack.g_del;
v_ins NUMBER := var_pack.g_ins;
v_upd NUMBER := var_pack.g_upd;
v_up_sal NUMBER := var_pack.g_up_sal;
BEGIN
IF v_del + v_ins + v_upd != 0 THEN
UPDATE audit_table SET
del = del + v_del, ins = ins + v_ins,
upd = upd + v_upd
WHERE user_name=USER AND tablename='EMPLOYEES'
AND column_name IS NULL;
END IF;
IF v_up_sal != 0 THEN
UPDATE audit_table SET upd = upd + v_up_sal
WHERE user_name=USER AND tablename='EMPLOYEES'
AND column_name = 'SALARY';
END IF;
-- resetting global variables in package VAR_PACK
var_pack.set_g_del (0); var_pack.set_g_ins (0);
var_pack.set_g_upd (0); var_pack.set_g_up_sal (0);
END audit_emp;
Demonstration: Using the
AUDIT_EMP Procedure
Updating the AUDIT_TABLE with the AUDIT_EMP Procedure
The AUDIT_EMP procedure updates the AUDIT_TABLE and calls the functions in the p
ackage VAR_PACK
that reset the package variables, ready for the next DML statement.
Oracle9i: Program with PL/SQL 16-40
16-40 Copyright Oracle Corporation, 2001. All rights reserved.
Procedure Package Trigger
xxxxxxxxxxxxxxxxxx
vvvvvvvvvvvvvvvvvv
xxxxxxxxxxxxxxxxxx
vvvvvvvvvvvvvvvvvv
xxxxxxxxxxxxxxxxxx
vvvvvvvvvvvvvvvvvv
xxxxxxxxxxxxxxxxxx
xxxxxxxxxxxxxxxxxx
vvvvvvvvvvvvvvvvvv
xxxxxxxxxxxxxxxxxx
vvvvvvvvvvvvvvvvvv
xxxxxxxxxxxxxxxxxx
vvvvvvvvvvvvvvvvvv
xxxxxxxxxxxxxxxxxx
Procedure A
declaration
Procedure B
definition
Summary
Procedure A
definition
Local
variable
Develop different types of procedural database constructs depending on their usa
ge.
Construct Usage
Procedure PL/SQL programming block that is stored in the database for repeated
execution
Package Group of related procedures, functions, variables, cursors, constants, a
nd
exceptions
Trigger PL/SQL programming block that is executed implicitly by a data manipulat
ion
statement
Oracle9i: Program with PL/SQL 16-41
16-41 Copyright Oracle Corporation, 2001. All rights reserved.
Practice 16 Overview
This practice covers the following topics:
Creating statement and row triggers
Creating advanced triggers to add to the
capabilities of the Oracle database
Practice 16 Overview
You create statement and row triggers in this practice. You create procedures th
at will be invoked from
the triggers.
Oracle9i: Program with PL/SQL 16-42
Practice 16
1. Changes to data are allowed on tables only during normal office hours of 8:45
a.m. until 5:30 p.m.,
Monday through Friday.
Create a stored procedure called SECURE_DML that prevents the DML statement from
executing
outside of normal office hours, returning the message, You may only make changes
during normal
office hours.
2. a. Create a statement trigger on the JOBS table that calls the above procedur
e.
b. Test the procedure by temporarily modifying the hours in the procedure and at
tempting to
insert a new record into the JOBS table. (Example: replace 08:45 with 16:45; Thi
s attempt results in
an error message)
After testing, reset the procedure hours asspecified in question 1 and recreate
the
procedure as in question 1 above.
If you have time:
3. Employees should receive an automatic increase in salary if the minimum salar
y for a job is
increased. Implement this requirement through a trigger on the JOBS table.
a. Create a stored procedure named UPD_EMP_SAL to update the salary amount. This
procedure accepts two parameters: the job ID for which salary has to be updated,
and the new
minimum salary for this job ID. This procedure is executed from the trigger on t
he JOBS
table.
b. Create a row trigger named UPDATE_EMP_SALARY on the JOBS table that invokes t
he
procedure UPD_EMP_SAL, when the minimum salary in the JOBS table is updated for
a
specified job ID.
c. Query the EMPLOYEES table to see the current salary for employees who are
programmers.
d. Increase the minimum salary for the Programmer job from 4,000 to 5,000.
e. Employee Lorentz (employee ID 107) had a salary of less than 4,500. Verify th
at her
salary has been increased to the new minimum of 5,000.
Instructor Note
If you encounter errors while performing practice question 3d, verify that you d
o not have the triggers
trig_constr_emp and constr_emp_trig in your schema. These triggers are created a
nd dropped
when you run the demo 16_trig_constr.sql and code example 16_34s.sql. Students d
id not
create these triggers earlier.

You might also like