`

触发器使用教程和命名规范

阅读更多
目  录  
1,触发器简介 1 
2,触发器示例 2 
3,触发器语法和功能  3 
4,例一:行级触发器之一    4 
5,例二:行级触发器之二    4 
6,例三:INSTEAD OF触发器  6 
7,例四:语句级触发器之一   8 
8,例五:语句级触发器之二   9 
9,例六:用包封装触发器代码  10 
10,触发器命名规范  11 
 
1,触发器简介  
触发器(Trigger)是数据库对象的一种,编码方式类似存储过程,与某张表(Table)相关联,当有DML语句对表进行操作时,可以引起触发器的执行,达到对插入记录一致性,正确性和规范性控制的目的。在当年C/S时代盛行的时候,由于客户端直接连接数据库,能保证数据库一致性的只有数据库本身,此时主键(Primary Key),外键(Foreign Key),约束(Constraint)和触发器成为必要的控制机制。而触发器的实现比较灵活,可编程性强,自然成为了最流行的控制机制。到了B/S时代,发展成4层架构,客户端不再能直接访问数据库,只有中间件才可以访问数据库。要控制数据库的一致性,既可以在中间件里控制,也可以在数据库端控制。很多的青睐Java的开发者,随之将数据库当成一个黑盒,把大多数的数据控制工作放在了Servlet中执行。这样做,不需要了解太多的数据库知识,也减少了数据库编程的复杂性,但同时增加了Servlet编程的工作量。从架构设计来看,中间件的功能是检查业务正确性和执行业务逻辑,如果把数据的一致性检查放到中间件去做,需要在所有涉及到数据写入的地方进行数据一致性检查。由于数据库访问相对于中间件来说是远程调用,要编写统一的数据一致性检查代码并非易事,一般采用在多个地方的增加类似的检查步骤。一旦一致性检查过程发生调整,势必导致多个地方的修改,不仅增加工作量,而且无法保证每个检查步骤的正确性。触发器的应用,应该放在关键的,多方发起的,高频访问的数据表上,过多使用触发器,会增加数据库负担,降低数据库性能。而放弃使用触发器,则会导致系统架构设计上的问题,影响系统的稳定性。  
 
 
2,触发器示例  
触发器代码类似存储过程,以PL/SQL脚本编写。下面是一个触发器的示例:  
新建员工工资表salary  
create table SALARY  
(  
  EMPLOYEE_ID NUMBER, --员工ID  
  MONTH       VARCHAR2(6), --工资月份  
  AMOUNT      NUMBER --工资金额  
)  
 
创建与salary关联的触发器salary_trg_rai  
1   Create or replace trigger salary_trg_rai  
2   After insert on salary  
3   For each row  
4   declare  
5   Begin  
6     Dbms_output.put_line(‘员工ID:’ || :new.employee_id);  
7     Dbms_output.put_line(‘工资月份:’ || :new.month);  
8     Dbms_output.put_line(‘工资:’ || :new.amount);  
9     Dbms_output.put_line(‘触发器已被执行’);  
10   End;  
打开一个SQL Window窗口(使用PL/SQL Developer工具),或在sqlplus中输入:  
Insert into salary(employee_id, month, amount) values(1, ‘200606’, 10000);  
执行后可以在sqlplus中,或在SQL Window窗口的Output中见到  
员工ID:1 
工资月份:200606 
工资:10000 
触发器已执行  
 
在代码的第一行,定义了数据库对象的类型是trigger,定义触发器的名称是salary_trg_rai  
第二行说明了这是一个after触发器,在DML操作实施之后执行。紧接着的insert说明了这是一个针对insert操作的触发器,每个对该表进行的insert操作都会执行这个触发器。  
第三行说明了这是一个针对行级的触发器,当插入的记录有n条时,在每一条插入操作时都会执行该触发器,总共执行n次。  
Declare后面跟的是本地变量定义部分,如果没有本地变量定义,此部分可以为空  
Begin和end括起来的代码,是触发器的执行部分,一般会对插入记录进行一致性检查,在本例中打印了插入的记录和“触发器已执行”。  
其中:new对象表示了插入的记录,可以通过:new.column_name来引用记录的每个字段值  
 
 
3,触发器语法和功能  
触发器的语法如下  
CREATE OR REPLACE TRIGGER trigger_name  
<before | after | instead of> <insert | update | delete> ON table_name  
[FOR EACH ROW]  
WHEN (condition)  
DECLARE  
BEGIN  
    --触发器代码  
END;  
 
Trigger_name是触发器的名称。<before | after | instead of>可以选择before或者after或instead of。Before表示在DML语句实施前执行触发器,而after表示在在dml语句实施之后执行触发器,instead of触发器用在对视图的更新上。<insert | update | delete>可以选择一个或多个DML语句,如果选择多个,则用or分开,如:insert or update。Table_name是触发器关联的表名。  
[FOR EACH ROW]为可选项,如果注明了FOR EACH ROW,则说明了该触发器是一个行级的触发器,DML语句处理每条记录都会执行触发器;否则是一个语句级的触发器,每个DML语句触发一次。  
WHEN后跟的condition是触发器的响应条件,只对行级触发器有效,当操作的记录满足condition时,触发器才被执行,否则不执行。Condition中可以通过new对象和old对象(注意区别于前面的:new和:old,在代码中引用需要加上冒号)来引用操作的记录。  
触发器代码可以包括三种类型:未涉及数据库事务代码,涉及关联表(上文语法中的table_name)数据库事务代码,涉及除关联表之外数据库事务代码。其中第一种类型代码只对数据进行简单运算和判断,没有DML语句,这种类型代码可以在所有的触发器中执行。第二种类型代码涉及到对关联表的数据操作,比如查询关联表的总记录数或者往关联表中插入一条记录,该类型代码只能在语句级触发器中使用,如果在行级触发器中使用,将会报ORA-04091错误。第三种类型代码涉及到除关联表之外的数据库事务,这种代码可以在所有触发器中使用。  
 
从触发器的功能上来看,可以分成3类:  
   重写列(仅限于before触发器)  
   采取行动(任何触发器)  
   拒绝事务(任何触发器)  
“重写列”用于对表字段的校验,当插入值为空或者插入值不符合要求,则触发器用缺省值或另外的值代替,在多数情况下与字段的default属性相同。这种功能只能在行级before触发器中执行。“采取行动”针对当前事务的特点,对相关表进行操作,比如根据当前表插入的记录更新其他表,银行中的总帐和分户帐间的总分关系就可以通过这种触发器功能来维护。“拒绝事务”用在对数据的合法性检验上,当更新的数据不满足表或系统的一致性要求,则通过抛出异常的方式拒绝事务,在其上层的代码可以捕获这个异常并进行相应操作。  
 
下面将通过举例说明,在例子中将触发器主体的语法一一介绍,读者可以在例子中体会触发器的功能。  
 
4,例一:行级触发器之一  
CREATE OR REPLACE TRIGGER salary_raiu  
AFTER INSERT OR UPDATE OF amount ON salary  
FOR EACH ROW  
BEGIN  
    IF inserting THEN  
        dbms_output.put_line(‘插入’);  
    ELSIF updating THEN  
dbms_output.put_line(‘更新amount列’);  
    END IF;  
END;  
以上是一个after insert和after update的行级触发器。在第二行中of amount on salary的意思是只有当amount列被更新时,update触发器才会有效。所以,以下语句将不会执行触发器:  
Update salary set month = ‘200601’ where month = ‘200606’;  
在触发器主体的if语句表达式中,inserting, updating和deleting可以用来区分当前是在做哪一种DML操作,可以作为把多个类似触发器合并在一个触发器中判别触发事件的属性。  
 
5,例二:行级触发器之二  
新建员工表employment  
CREATE TABLE EMPLOYMENT  
(  
  EMPLOYEE_ID NUMBER, --员工ID  
  MAXSALARY   NUMBER --工资上限  
)  
插入两条记录  
Insert into employment values(1, 1000);  
Insert into employment values(2, 2000);  
 
CREATE OR REPLACE TRIGGER salary_raiu  
AFTER INSERT OR UPDATE OF amount ON salary  
FOR EACH ROW  
WHEN ( NEW.amount >= 1000 AND (old.amount IS NULL OR OLD.amount <= 500))  
DECLARE  
    v_maxsalary NUMBER;  
BEGIN  
    SELECT maxsalary  
        INTO v_maxsalary  
        FROM employment  
     WHERE employee_id = :NEW.employee_id;  
    IF :NEW.amount > v_maxsalary THEN  
        raise_application_error(-20000, '工资超限');  
    END IF;  
END;  
 
以上的例子引入了一个新的表employment,表中的maxsalary字段代表该员工每月所能分配的最高工资。下面的触发器根据插入或修改记录的employee_id,在employment表中查到该员工的每月最高工资,如果插入或修改后的amount超过这个值,则报错误。  
代码中的when子句表明了该触发器只针对修改或插入后的amount值超过1000,而修改前的amount值小于500的记录。New对象和old对象分别表示了操作前和操作后的记录对象。对于insert操作,由于当前操作记录无历史对象,所以old对象中所有属性是null;对于delete操作,由于当前操作记录没有更新对象,所以new对象中所有属性也是null。但在这两种情况下,并不影响old和new对象的引用和在触发器主体中的使用,和普通的空值作同样的处理。  
在触发器主体中,先通过:new.employee_id,得到该员工的工资上限,然后在if语句中判断更新后的员工工资是否超限,如果超限则错误代码为-20000,错误信息为“工资超限”的自定义错误。其中的raise_application_error包含两个参数,前一个是自定义错误代码,后一个是自定义错误代码信息。其中自定义错误代码必须小于或等于-20000。执行完该语句后,一个异常被抛出,如果在上一层有exception子句,该异常将被捕获。如下面代码:  
DECLARE  
    code NUMBER;  
    msg  VARCHAR2(500);  
BEGIN  
    INSERT INTO salary (employee_id, amount) VALUES (2, 5000);  
EXCEPTION  
    WHEN OTHERS THEN  
        code := SQLCODE;  
        msg  := substr(SQLERRM, 1, 500);  
        dbms_output.put_line(code);  
        dbms_output.put_line(msg);  
END;  
执行后,将在output中或者sqlplus窗口中见着以下信息:  
-20000 
ORA-20000: 工资超出限制  
ORA-06512: 在"SCOTT.SALARY_RAI", line 9 
ORA-04088: 触发器 'SCOTT.SALARY_RAI' 执行过程中出错  
 
这里的raise_application_error相当于拒绝了插入或者修改事务,当上层代码接受到这个异常后,判断该异常代码等于-20000,可以作出回滚事务或者继续其他事务的处理。  
 
以上两个例子中用到的inserting, updating, deleting和raise_application_error都是dbms_standard包中的函数,具体的说明可以参照Oracle的帮助文档。  
create or replace package sys.dbms_standard is  
  procedure raise_application_error(num binary_integer, msg varchar2,  
  function inserting return boolean;  
  function deleting  return boolean;  
  function updating  return boolean;  
  function updating (colnam varchar2) return boolean;  
end;  
 
对于before和after行级触发器,:new和:old对象的属性值都是一样的,主要是对于在Oracle约束(Constraint)之前或之后的执行触发器的选择。需要注意的是,可以在before行触发器中更改:new对象中的值,但是在after行触发器就不行。  
 
下面介绍一种instead of触发器,该触发器主要使用在对视图的更新上,以下是instead of触发器的语法:  
CREATE OR REPLACE TRIGGER trigger_name  
INSTEAD OF <insert | update | delete> ON view_name  
[FOR EACH ROW]  
WHEN (condition)  
DECLARE  
BEGIN  
    --触发器代码  
END;  
 
其他部分语法同前面所述的before和after语法是一样的,唯一不同的是在第二行用上了instead of关键字。对于普通的视图来说,进行insert等操作是被禁止的,因为Oracle无法知道操作的字段具体是哪个表中的字段。但我们可以通过建立instead of触发器,在触发器主体中告诉Oracle应该更新,删除或者修改哪些表的哪部分字段。如:  
 
6,例三:instead of触发器  
新建视图  
CREATE VIEW employee_salary(employee_id, maxsalary, MONTH, amount) AS   
SELECT a.employee_id, a.maxsalary, b.MONTH, b.amount  
FROM employment a, salary b  
WHERE a.employee_id = b.employee_id  
 
如果执行插入语句  
INSERT INTO employee_salary(employee_id, maxsalary, MONTH, amount)  
VALUES(10, 100000, '200606', 10000);  
系统会报错:  
ORA-01779:无法修改与非键值保存表对应的列  
 
我们可以通过建立以下的instead of存储过程,将插入视图的值分别插入到两个表中:  
create or replace trigger employee_salary_rii  
  instead of insert on employee_salary    
  for each ROW  
DECLARE  
    v_cnt NUMBER;  
BEGIN  
  --检查是否存在该员工信息  
    SELECT COUNT(*)  
        INTO v_cnt  
        FROM employment  
     WHERE employee_id = :NEW.employee_id;  
    IF v_cnt = 0 THEN  
        INSERT INTO employment  
            (employee_id, maxsalary)  
        VALUES  
            (:NEW.employee_id, :NEW.maxsalary);  
    END IF;  
  --检查是否存在该员工的工资信息  
    SELECT COUNT(*)  
        INTO v_cnt  
        FROM salary  
     WHERE employee_id = :NEW.employee_id  
         AND MONTH = :NEW.MONTH;  
    IF v_cnt = 0 THEN  
        INSERT INTO salary  
            (employee_id, MONTH, amount)  
        VALUES  
            (:NEW.employee_id, :NEW.MONTH, :NEW.amount);  
    END IF;  
END employee_salary_rii;  
 
该触发器被建立后,执行上述insert操作,系统就会提示成功插入一条记录。  
但需要注意的是,这里的“成功插入一条记录”,只是Oracle并未发现触发器中有异常抛出,而根据insert语句中涉及的记录数作出一个判断。若触发器的主体什么都没有,只是一个空语句,Oracle也会报“成功插入一条记录”。同样道理,即使在触发器主体里往多个表中插入十条记录,Oracle的返回也是“成功插入一条记录”。  
 
 
 
 
行级触发器可以解决大部分的问题,但是如果需要对本表进行扫描检查,比如要检查总的工资是否超限了,用行级触发器是不行的,因为行级触发器主体中不能有涉及到关联表的事务,这时就需要用到语句级触发器。以下是语句级触发器的语法:  
CREATE OR REPLACE TRIGGER trigger_name  
<before | after | instead of ><insert | update | delete > ON table_name  
DECLARE  
BEGIN  
    --触发器主体  
END;  
 
从语法定义上来看,行级触发器少了for each row,也不能使用when子句来限定入口条件,其他部分都是一样的,包括insert, update, delete和instead of都可以使用。  
 
 
7,例四:语句级触发器之一  
CREATE OR REPLACE TRIGGER salary_saiu  
AFTER INSERT OR UPDATE OF amount ON salary  
DECLARE  
    v_sumsalary NUMBER;  
BEGIN  
  SELECT SUM(amount) INTO v_sumsalary FROM salary;  
    IF v_sumsalary > 500000 THEN  
        raise_application_error(-20001, '总工资超过500000');  
    END IF;  
END;  
 
以上代码定义了一个语句级触发器,该触发器检查在insert和update了amount字段后操作后,工资表中所有工资记录累加起来是否超过500000,如果超过则抛出异常。从这个例子可以看出,语句级触发器可以对关联表表进行扫描,扫描得到的结果可以用来作为判断一致性的标志。需要注意的是,在before语句触发器主体和after语句触发器主体中对关联表进行扫描,结果是不一样的。在before语句触发器主体中扫描,扫描结果将不包括新插入和更新的记录,也就是说当以上代码换成 before触发器后,以下语句将不报错:  
INSERT INTO salary(employee_id, month, amount) VALUEs(2, '200601', 600000)  
这是因为在主体中得到的v_sumsalary并不包括新插入的600000工资。  
另外,在语句级触发器中不能使用:new和:old对象,这一点和行级触发器是显著不同的。如果需要检查插入或更新后的记录,可以采用临时表技术。  
临时表是一种Oracle数据库对象,其特点是当创建数据的进程结束后,进程所创建的数据也随之清除。进程与进程不可以互相访问同一临时表中对方的数据,而且对临时表进行操作也不产生undo日志,减少了数据库的消耗。具体有关临时表的知识,可以参看有关书籍。  
为了在语句级触发器中访问新插入后修改后的记录,可以增加行级触发器,将更新的记录插入临时表中,然后在语句级触发器中扫描临时表,获得修改后的记录。临时表的表结构一般与关联表的结构一致。  
 
 
8,例五:语句级触发器之二  
目的:限制每个员工的总工资不能超过50000,否则停止对该表操作。  
创建临时表  
create global temporary table SALARY_TMP  
(  
  EMPLOYEE_ID NUMBER,  
  MONTH       VARCHAR2(6),  
  AMOUNT      NUMBER  
)  
on commit delete rows;  
 
为了把操作记录插入到临时表中,创建行级触发器:  
CREATE OR REPLACE TRIGGER salary_raiu  
AFTER INSERT OR UPDATE OF amount ON salary  
FOR EACH ROW  
BEGIN  
  INSERT INTO salary_tmp(employee_id, month, amount)  
  VALUES(:NEW.employee_id, :NEW.MONTH, :NEW.amount);  
END;  
该触发器的作用是把更新后的记录信息插入到临时表中,如果更新了多条记录,则每条记录都会保存在临时表中。  
 
创建语句级触发器:  
CREATE OR REPLACE TRIGGER salary_sai  
AFTER INSERT OR UPDATE OF amount ON salary  
DECLARE  
    v_sumsalary NUMBER;  
BEGIN  
    FOR cur IN (SELECT * FROM salary_tmp) LOOP  
        SELECT SUM(amount)  
            INTO v_sumsalary  
            FROM salary  
         WHERE employee_id = cur.employee_id;  
        IF v_sumsalary > 50000 THEN  
            raise_application_error(-20002, '员工累计工资超过50000');  
        END IF;  
    DELETE FROM salary_tmp;  
    END LOOP;  
END;  
 
该触发器首先用游标从salary_tmp临时表中逐条读取更新或插入的记录,取employee_id,在关联表salary中查找所有相同员工的工资记录,并求和。若某员工工资总和超过50000,则抛出异常。如果检查通过,则清空临时表,避免下次检查相同的记录。  
执行以下语句:  
INSERT INTO salary(employee_id, month, amount) VALUEs(7, '200601', 20000);  
INSERT INTO salary(employee_id, month, amount) VALUEs(7, '200602', 20000);  
INSERT INTO salary(employee_id, month, amount) VALUEs(7, '200603', 20000);  
在执行第三句时系统报错:  
ORA-20002:员工累计工资超过50000 
查询salary表,发现前两条记录正常插入了,第三条记录没有插入。  
 
 
如果系统结构比较复杂,而且触发器的代码比较多,在触发器主体中写过多的代码,对于维护来说是一个困难。这时可以将所有触发器的代码写到同一个包中,不同的触发器代码以不同的存储过程封装,然后触发器主体中调用这部分代码。  
 
9,例六:用包封装触发器代码  
目的:改写例五,封装触发器主体代码  
创建代码包:  
CREATE OR REPLACE PACKAGE BODY salary_trigger_pck IS  
 
    PROCEDURE load_salary_tmp(i_employee_id IN NUMBER,  
                            i_month       IN VARCHAR2,  
                            i_amount      IN NUMBER) IS  
    BEGIN  
        INSERT INTO salary_tmp VALUES (i_employee_id, i_month, i_amount);  
    END load_salary_tmp;  
 
    PROCEDURE check_salary IS  
        v_sumsalary NUMBER;  
    BEGIN  
        FOR cur IN (SELECT * FROM salary_tmp) LOOP  
            SELECT SUM(amount)  
                INTO v_sumsalary  
                FROM salary  
             WHERE employee_id = cur.employee_id;  
            IF v_sumsalary > 50000 THEN  
                raise_application_error(-20002, '员工累计工资超过50000');  
            END IF;  
            DELETE FROM salary_tmp;  
        END LOOP;  
    END check_salary;  
END salary_trigger_pck;  
包salary_trigger_pck中有两个存储过程,load_salary_tmp用于在行级触发器中调用,往salary_tmp临时表中装载更新或插入记录。而check_salary用于在语句级触发器中检查员工累计工资是否超限。  
 
修改行级触发器和语句级触发器:  
CREATE OR REPLACE TRIGGER salary_raiu  
    AFTER INSERT OR UPDATE OF amount ON salary  
    FOR EACH ROW  
BEGIN  
    salary_trigger_pck.load_salary_tmp(:NEW.employee_id,     :NEW.MONTH, :NEW.amount);  
END;  
 
CREATE OR REPLACE TRIGGER salary_sai  
AFTER INSERT OR UPDATE OF amount ON salary  
BEGIN  
    salary_trigger_pck.check_salary;  
END;  
 
这样主要代码就集中到了salary_trigger_pck中,触发器主体中只实现了一个调用功能。  
 
10,触发器命名规范  
为了方便对触发器命名和根据触发器名称了解触发器含义,需要定义触发器的命名规范:  
Trigger_name = table_name_trg_<R|S><A|B|I><I|U|D>  
 
触发器名限于30个字符。必须缩写表名,以便附加触发器属性信息。  
<R|S>基于行级(row)还是语句级(statement)的触发器  
<A|B|I>after, before或者是instead of触发器  
<I|U|D>触发事件是insert,update还是delete。如果有多个触发事件则连着写  
 
例如:  
Salary_rai      salary表的行级after触发器,触发事件是insert  
Employee_sbiud  employee表的语句级before触发器,触发事件是insert,update和delete
分享到:
评论

相关推荐

    oracle触发器使用,很详细

    触发器使用教程和命名规范 1 1,触发器简介 1 2,触发器示例 2 3,触发器语法和功能 3 4,例一:行级触发器之一 4 5,例二:行级触发器之二 4 6,例三:INSTEAD OF触发器 6 7,例四:语句级触发器之一 8 8,...

    oracle各种文档

    数据仓库建模技术.pdf 在数据库中实现base64编码和解码.doc 触发器使用教程和命名规范.doc TransactSQL.doc Schema常用脚本.doc rsultset.doc oracle傻瓜手册 ORACLE 中存储过程定期分割表.doc ORACLE 物理文件大小...

    20篇oracle的学习文章

    "触发器使用教程和命名规范.doc"提供了关于触发器的使用指导,触发器是数据库中的一种存储过程,可以在特定事件(如INSERT、UPDATE或DELETE)发生时自动执行。同时,良好的命名规范有助于代码的可读性和维护性。 ...

    华为 Verilog_HDL_三套教程

    本系列教程聚焦于华为对于Verilog HDL的使用规范、基础学习以及实际电路设计,旨在帮助工程师提升在数字集成电路设计中的技能。 首先,我们来看"华为 Verilog HDL代码书写规范"。这个部分将详细介绍华为在编写...

    PL/SQL(函数、包、触发器、异常、游标等)

    包的使用能提高代码的复用性和组织性,同时减少命名冲突。 3. **触发器**:触发器是一种特殊的存储过程,它在数据库发生特定事件(如INSERT、UPDATE或DELETE)时自动执行。触发器可以用于强制业务规则、维护参照...

    MySQL ER设计工具 Workbench的使用教程

    在MySQL Workbench中,还可以设定字段的排序规则(Collation)和其它高级选项,如触发器(Trigger)和分区(Partitioning)。 在设计表时,还可以设定主键、外键以及默认值等属性。主键用于唯一标识表中的每条记录...

    SQL2000超详细图文教程

    2. **SQL命名规则和注释**:教授如何遵循SQL的命名规范,以及如何在代码中添加注释,提高代码可读性。 3. **变量**:包括全局变量和局部变量的声明和使用。 4. **运算符**:涵盖算术、赋值、位、字符串串联、比较和...

    PowerDesigner使用教程

    在使用PowerDesigner时,应注意遵循良好的建模规范,如合理命名、保持模型清晰、适当地使用注释等,以便于理解和维护。 通过本教程,你将学习如何利用PowerDesigner进行数据库设计,从创建概念模型开始,逐步细化...

    MySQl-Workbench使用教程

    此外,可以设定表的其他物理属性,如字符集(Collation),甚至高级选项如触发器(Trigger)和分区(Partitioning)。 - **添加额外字段**:继续在`ORDER`表中增加`ORDER_DATE`和`ORDER_TYPE`字段。`ORDER_TYPE`...

    PHP6与MySQL5基础教程第三版源代码(09.11.25更新)

    MySQL5,是广泛使用的开源关系型数据库管理系统,其主要特性包括事务处理、存储过程、触发器、视图等。这些特性使得MySQL5在处理复杂的数据操作和并发控制时表现出色。学习MySQL5,你需要理解SQL语言的基本语法,...

    数据库参考教程-数据库设计规范4net.xlsx

    ,S012、S013、S014,,,,, D007,触发器,慎用,如有使用必须说明原因,,S015、S016、S017,,,,, D008,存储过程,设计者应将常用的底层数据访问功能封装为存储过程、函数,并说明他们的职责、输入输出参数、重点逻辑,以...

    数据库教程视频第三节.rar

    在第二节的内容回顾中,我们探讨了SQL Server的命名规则,这是在创建数据库对象时必须遵循的重要规范,包括表、视图、存储过程等,以确保名称的唯一性和可读性。 接着,我们了解了数据库的基本概念。数据库是存储和...

    Oracle 编程自编教程

    【Oracle编程自编教程】是一...通过这份教程,学习者将掌握Oracle数据库的使用,包括编写和管理PL/SQL代码,创建和操作数据库对象,以及理解如何利用工具提升开发效率。这为数据库管理和应用程序开发打下了坚实的基础。

    【自整理】PL SQL 入门教程

    - 支持使用标号和GOTO语句来跳转到PL/SQL块内的特定位置。 - **3.5 NULL语句** - 当需要一个空的操作时使用。 #### 第四章:游标 - **4.1 游标概念** - **4.1.1 处理显式游标** - 显式游标用于处理查询结果集...

    oracle基础教程

    理解Oracle中的主要数据库对象至关重要,包括表格、视图、索引、存储过程、触发器和游标等。每个对象都有其特定的功能和使用场景,比如表格用于存储数据,视图提供定制化的数据视图,索引提高查询速度。 五、PL/SQL...

    plsql学习教程

    - 代码规范:遵循命名约定,保持代码整洁,注释清晰,提高代码可读性。 - 错误处理:确保每个可能出错的地方都有适当的异常处理。 - 性能优化:避免不必要的数据访问,合理使用索引,减少嵌套循环,考虑使用绑定...

    Verilog HDL 华为入门教程.

    - **命名规范**:清晰、有意义的变量和模块命名有助于理解和维护。 - **代码组织**:模块化、层次化的代码结构,避免全局变量,提高可维护性。 9. **FPGA应用** - **FPGA基础**:了解FPGA的工作原理,如查找表...

    EDA技术实用教程第4版VERILOG课件第3章

    - **规范的程序书写格式**:为了提高代码的可读性和维护性,Verilog代码应该遵循一定的书写规范,如缩进、空格的使用等。 - **文件取名和存盘**:文件名的选择应符合命名规则,并且最好能够反映文件的内容。 ##### ...

Global site tag (gtag.js) - Google Analytics