行级DML触发器
每当一条记录出现更新操作时进行触发操作定义时要定义FOR EACH ROW
使用":old.字段"和":new.字段"标识符
No. | 触发语句 | :old字段 | :new字段 |
1 | INSERT | 未定义,字段内容为NULL | INSERT操作结束后,为增加数据值 |
2 | UPDATE | 更新数据前的原始值 | UPDATE操作之后,更新数据后的新值 |
3 | DELETE | 删除前的原始值 | 未定义,字段内容均为NULL |
":old.字段"和":new.字段只对行级触发有效,如果触发器没有定义FOR EACH RO则无效而报语法错误
示例一、增加员工信息时,其职位必须在已经职位内选择,并且工资不能超过5000,入职日期为当前日期
--创建触发器 create or replace trigger myempinsertbefore insert on myemp for each ROWdeclarev_date DATE;v_jobcount NUMBER; BEGINSELECT COUNT(empno) INTO v_jobcount FROM myemp WHERE:new.job IN(SELECT DISTINCT job FROM myemp);IF v_jobcount=0 THEN --没有找到此职位raise_application_error(-20009,'职位错误');ELSEIF :new.Sal>5000 THENraise_application_error(-20003,'工资不能超过5000');END IF;END IF;:new.Hiredate:=SYSDATE; end myempinsert; --调用执行 DECLARE BEGIN--添加错误信息 INSERT INTO myemp(empno,ename,job,mgr,sal,deptno)VALUES(9999,'Bdqn','MNAGER',7788,8000,10); EXCEPTIONWHEN OTHERS THENdbms_output.put_line(SQLERRM); END;
示例二、myemp工资涨幅不能超过10%
--创建触发器 CREATE OR REPLACE TRIGGER MYEMPADDSAL_TRIGGERBEFORE UPDATE ON MYEMPFOR EACH ROW DECLAREBEGINIF ABS((:NEW.SAL - :OLD.SAL) / :OLD.SAL) > 0.1 THENraise_application_error(-20005,'工资最大涨幅不能超过10%');END IF;END MYEMPADDSAL_TRIGGER;--调用 DECLARE BEGIN--添加错误信息 --INSERT INTO myemp(empno,ename,job,mgr,sal,deptno)VALUES(9999,'Bdqn','MNAGER',7788,8000,10); UPDATE myemp SET sal=3000 WHERE empno=7369; EXCEPTIONWHEN OTHERS THENdbms_output.put_line(SQLERRM); END;
示例三、使用序列
--创建表 CREATE TABLE mt(ID NUMBER,NAME VARCHAR2(30),address varchar2(50),CONSTRAINT pd_mtid PRIMARY KEY(ID) ); SELECT * FROM mt; --创建序列 CREATE SEQUENCE mt_se; --创建触发器 create or replace trigger mt_triggerbefore insert on mt for each row declare-- local variables here beginSELECT mt_se.NEXTVAL INTO :new.id FROM dual; end mt_trigger; --插入数据 INSERT INTO mt(name,address)VALUES('张三','深圳'); INSERT INTO mt(name,address)VALUES('李四','深圳'); SELECT * FROM mt; --发现ID自动增加
12c自动序列
oracle12C中可以用以下写法:更直观
--创建表 CREATE TABLE mt(ID NUMBER,NAME VARCHAR2(30),address varchar2(50),CONSTRAINT pd_mtid PRIMARY KEY(ID) ); CREATE TABLE mtme AS SELECT * FROM mt WHERE 1=2; SELECT * FROM mt; SELECT * FROM mtme; --创建序列 CREATE SEQUENCE mt_se; --创建触发器 create or replace trigger mtmebefore insert on mtme for each row declareBEGINDELETE mtme;INSERT INTO mt(id,name,address)VALUES(mt_se.nextval,:NEW.NAME,:NEW.ADDRESS); end mtme;--插入数据 INSERT INTO mt(name,address)VALUES('张三','深圳'); INSERT INTO mt(name,address)VALUES('李四','深圳');