PL/SQL块结构和组成元素

2.1 PL/SQL块

PL/SQL程序由三个块组成,即声明部分、执行部分、异常处理部分。
PL/SQL块的结构如下:
DECLARE
—声明部分: 在此声明PL/SQL用到的变量,类型及游标,以及局部的存储过程和函数
BEGIN
— 执行部分: 过程及SQL 语句 , 即程序的主要部分
EXCEPTION
— 执行异常部分: 错误处理
END;
其中:执行部分不能省略。
PL/SQL块可以分为以下几类:
1. 无名块或匿名块(anonymous):动态构造,只能执行一次,可调用其它程序,但不能被其它程序调用。
2. 命名块(named):是带有名称的匿名块,这个名称就是标签。
3. 子程序(subprogram):存储在数据库中的存储过程、函数等。当在数据库上建立好后可以在其它程序中调用它们。
4. 触发器(Trigger):当数据库发生操作时,会触发一些事件,从而自动执行相应的程序。
5. 程序包/包(package):存储在数据库中的一组子程序、变量定义。在包中的子程序可以被其它程序包或子程序调用。但如果声明的是局部子程序,则只能在定义该局部子程序的块中调用该局部子程序。

2.2 PL/SQL结构

1.PL/SQL块中可以包含子块;
2.子块可以位于 PL/SQL中的任何部分;
3.子块也即PL/SQL中的一条命令;

2.3 标识符

PL/SQL程序设计中的标识符定义与SQL 的标识符定义的要求相同。要求和限制有:
1.标识符名不能超过30字符;
2.第一个字符必须为字母;
3.不分大小写;
4.不能用’-‘(减号);
5.不能是SQL保留字。
提示: 一般不要把变量名声明与表中字段名完全一样,如果这样可能得到不正确的结果.
例如:下面的例子将会删除所有的纪录,而不是’EricHu’的记录;

  1. DECLARE
  2. ename varchar2(20) :='EricHu';
  3. BEGIN
  4. DELETE FROM scott.emp WHERE ename=ename;
  5. END;
  1. 变量命名在PL/SQL中有特别的讲究,建议在系统的设计阶段就要求所有编程人员共同遵守一定的要求,使得整个系统的文档在规范上达到要求。下面是建议的命名方法:

2.4 PL/SQL 变量类型

在前面的介绍中,有系统的数据类型,也可以自定义数据类型。下表给出ORACLE类型和PL/SQL中的变量类型的合法使用列表:

2.4.1 变量类型

在ORACLE8i中可以使用的变量类型有:

例1. 插入一条记录并显示;

 DECLARE
    Row_id ROWID;
    info    VARCHAR2(40);
 BEGIN
 INSERT INTO scott.dept VALUES (90, '财务室', '海口')
 RETURNING rowid, dname||':'||to_char(deptno)||':'||loc
 INTO row_id, info;
 DBMS_OUTPUT.PUT_LINE('ROWID:'||row_id);
 DBMS_OUTPUT.PUT_LINE(info);
 END;

其中:
RETURNING子句用于检索INSERT语句中所影响的数据行数,当INSERT语句使用VALUES 子句插入数据时,RETURNING 字句还可将列表达式、ROWID和REF值返回到输出变量中。在使用RETURNING 子句是应注意以下几点限制:
1.不能与DML语句和远程对象一起使用;
2.不能检索LONG 类型信息;
3.当通过视图向基表中插入数据时,只能与单基表视图一起使用。
例2. 修改一条记录并显示

 DECLARE
    Row_id ROWID;
    info   VARCHAR2(40);
 BEGIN
   UPDATE dept SET deptno=100 WHERE DNAME='财务室'
   RETURNING rowid, dname||':'||to_char(deptno)||':'||loc
   INTO row_id, info;
   DBMS_OUTPUT.PUT_LINE('ROWID:'||row_id);
   DBMS_OUTPUT.PUT_LINE(info);
 END;

其中:
RETURNING子句用于检索被修改行的信息。当UPDATE语句修改单行数据时,RETURNING 子句可以检索被修改行的ROWID和REF值,以及行中被修改列的列表达式,并可将他们存储到PL/SQL变量或复合变量中;当UPDATE语句修改多行数据时,RETURNING 子句可以将被修改行的ROWID和REF值,以及列表达式值返回到复合变量数组中。在UPDATE中使用RETURNING 子句的限制与INSERT语句中对RETURNING子句的限制相同。
例3. 删除一条记录并显示

 DECLARE
    Row_id ROWID;
    info   VARCHAR2(40);
 BEGIN
   DELETE dept WHERE DNAME='办公室'
   RETURNING rowid, dname||':'||to_char(deptno)||':'||loc
   INTO row_id, info;
   DBMS_OUTPUT.PUT_LINE('ROWID:'||row_id);
   DBMS_OUTPUT.PUT_LINE(info);
 END;

其中:
RETURNING子句用于检索被删除行的信息:当DELETE语句删除单行数据时,RETURNING 子句可以检索被删除行的ROWID和REF值,以及被删除列的列表达式,并可将他们存储到PL/SQL变量或复合变量中;当DELETE语句删除多行数据时,RETURNING 子句可以将被删除行的ROWID和REF值,以及列表达式值返回到复合变量数组中。在DELETE中使用RETURNING 子句的限制与INSERT语句中对RETURNING子句的限制相同。

2.4.2 复合类型

ORACLE 在 PL/SQL 中除了提供象前面介绍的各种类型外,还提供一种称为复合类型的类型---记录和表.<br />2.4.2.1 记录类型<br />记录类型类似于C语言中的结构数据类型,它把逻辑相关的、分离的、基本数据类型的变量组成一个整体存储起来,它必须包括至少一个标量型或RECORD数据类型的成员,称作PL/SQL RECORD 的域(FIELD),其作用是存放互不相同但逻辑相关的信息。在使用记录数据类型变量时,需要先在声明部分先定义记录的组成、记录的变量,然后在执行部分引用该记录变量本身或其中的成员。<br />定义记录类型语法如下:<br />TYPE record_name IS RECORD( <br />   v1 data_type1  [NOT NULL]  [:= default_value ], <br />   v2 data_type2  [NOT NULL]  [:= default_value ], <br />   ...... <br />   vn data_typen  [NOT NULL]  [:= default_value ] );<br />可以用 SELECT语句对记录变量进行赋值,只要保证记录字段与查询结果列表中的字段相配即可。<br />例 :
 DECLARE
 --定义与hr.employees表中的这几个列相同的记录数据类型
    TYPE RECORD_TYPE_EMPLOYEES IS RECORD(
         f_name   hr.employees.first_name%TYPE,
         h_date   hr.employees.hire_date%TYPE,
         j_id     hr.employees.job_id%TYPE);
 --声明一个该记录数据类型的记录变量
    v_emp_record RECORD_TYPE_EMPLOYEES;
 BEGIN
    SELECT first_name, hire_date, job_id INTO v_emp_record
    FROM employees
    WHERE employee_id = &emp_id;
    DBMS_OUTPUT.PUT_LINE('雇员名称:'||v_emp_record.f_name
              ||'  雇佣日期:'||v_emp_record.h_date
              ||'  岗位:'||v_emp_record.j_id);
 END;

一个记录类型的变量只能保存从数据库中查询出的一行记录,若查询出了多行记录,就会出现错误。
2.4.2.2 数组类型
数据是具有相同数据类型的一组成员的集合。每个成员都有一个唯一的下标,它取决于成员在数组中的位置。在PL/SQL中,数组数据类型是VARRAY。
定义VARRY数据类型语法如下:
TYPE varray_name IS VARRAY(size) OF element_type [NOT NULL];
varray_name是VARRAY数据类型的名称,size是下整数,表示可容纳的成员的最大数量,每个成员的数据类型是element_type。默认成员可以取空值,否则需要使用NOT NULL加以限制。对于VARRAY数据类型来说,必须经过三个步骤,分别是:定义、声明、初始化。
例6 :

 DECLARE
 --定义一个最多保存5个VARCHAR(25)数据类型成员的VARRAY数据类型
    TYPE reg_varray_type IS VARRAY(5) OF VARCHAR(25);
 --声明一个该VARRAY数据类型的变量
    v_reg_varray REG_VARRAY_TYPE;
 BEGIN
 --用构造函数语法赋予初值
    v_reg_varray := reg_varray_type
          ('中国', '美国', '英国', '日本', '法国');
    DBMS_OUTPUT.PUT_LINE('地区名称:'||v_reg_varray(1)||'、'
                                     ||v_reg_varray(2)||'、'
                                     ||v_reg_varray(3)||'、'
                                     ||v_reg_varray(4));
    DBMS_OUTPUT.PUT_LINE('赋予初值NULL的第5个成员的值:'||v_reg_varray(5));
 --用构造函数语法赋予初值后就可以这样对成员赋值
    v_reg_varray(5) := '法国';
    DBMS_OUTPUT.PUT_LINE('第5个成员的值:'||v_reg_varray(5));
 END;

2.4.2.3 使用%TYPE
定义一个变量,其数据类型与已经定义的某个数据变量(尤其是表的某一列)的数据类型相一致,这时可以使用%TYPE。
使用%TYPE特性的优点在于:
1.所引用的数据库列的数据类型可以不必知道;
2.所引用的数据库列的数据类型可以实时改变,容易保持一致,也不用修改PL/SQL程序。
例7:
1 DECLARE
2 — 用%TYPE 类型定义与表相配的字段
3 TYPE T_Record IS RECORD(
4 T_no emp.empno%TYPE,
5 T_name emp.ename%TYPE,
6 T_sal emp.sal%TYPE );
7 — 声明接收数据的变量
8 v_emp T_Record;
9 BEGIN
10 SELECT empno, ename, sal INTO v_emp FROM emp WHERE empno=7788;
11 DBMS_OUTPUT.PUT_LINE
12 (TO_CHAR(v_emp.t_no)||’ ‘||v_emp.t_name||’ ‘ || TO_CHAR(v_emp.t_sal));
13 END;
2.4.3 使用%ROWTYPE
PL/SQL 提供%ROWTYPE操作符, 返回一个记录类型, 其数据类型和数据库表的数据结构相一致。
使用%ROWTYPE特性的优点在于:
l 所引用的数据库中列的个数和数据类型可以不必知道;
l 所引用的数据库中列的个数和数据类型可以实时改变,容易保持一致,也不用修改PL/SQL程序。
例9:
1 DECLARE
2 v_empno emp.empno%TYPE :=&no;
3 rec emp%ROWTYPE;
4 BEGIN
5 SELECT * INTO rec FROM emp WHERE empno=v_empno;
6 DBMS_OUTPUT.PUT_LINE(‘姓名:’||rec.ename||’工资:’||rec.sal||’工作时间:’||rec.hiredate);
7 END;

2.4.4 LOB类型

ORACLE提供了LOB (Large OBject)类型,用于存储大的数据对象的类型。ORACLE目前主要支持BFILE, BLOB, CLOB 及 NCLOB 类型。<br />BFILE (Movie)<br />    存放大的二进制数据对象,这些数据文件不放在数据库里,而是放在操作系统的某个目录里,数据库的表里只存放文件的目录。<br />BLOB(Photo)<br />    存储大的二进制数据类型。变量存储大的二进制对象的位置。大二进制对象的大小<=4GB。<br />CLOB(Book)<br />    存储大的字符数据类型。每个变量存储大字符对象的位置,该位置指到大字符数据块。大字符对象的大小<=4GB。<br />NCLOB<br />    存储大的NCHAR字符数据类型。每个变量存储大字符对象的位置,该位置指到大字符数据块。大字符对象的大小<=4GB。

2.4.5 BIND 变量

绑定变量是在主机环境中定义的变量。在PL/SQL 程序中可以使用绑定变量作为他们将要使用的其它变量。为了在PL/SQL 环境中声明绑定变量,使用命令VARIABLE。例如:
VARIABLE return_code NUMBER
VARIABLE return_msg VARCHAR2(20)
可以通过SQLPlus命令中的PRINT 显示绑定变量的值。例如:
PRINT return_code
PRINT return_msg
例10:
1 VARIABLE result NUMBER;
2 BEGIN
3 SELECT (sal
10)+nvl(comm, 0) INTO :result FROM emp
4 WHERE empno=7844;
5 END;
6 —然后再执行
7 PRINT result

2.4.6 PL/SQL 表(TABLE)

定义记录表(或索引表)数据类型。它与记录类型相似,但它是对记录类型的扩展。它可以处理多行记录,类似于高级中的二维数组,使得可以在PL/SQL中模仿数据库中的表。
定义记录表类型的语法如下:
TYPE table_name IS TABLE OF element_type [NOT NULL]
INDEX BY [BINARY_INTEGER | PLS_INTEGER | VARRAY2];
关键字INDEX BY表示创建一个主键索引,以便引用记录表变量中的特定行。

例11:
1 DECLARE
2 TYPE depttable_type IS TABLE OF
3 dept%ROWTYPE INDEX BY BINARY_INTEGER;
4 my_dname_table dept_table_type;
5 v_count number(2) :=4;
6 BEGIN
7 FOR int IN 1 .. v_count LOOP
8 SELECT
INTO my_dname_table(int) FROM dept WHERE deptno=int_10;
9 END LOOP;
10 FOR int IN my_dname_table.FIRST .. my_dname_table.LAST LOOP
11 DBMS_OUTPUT.PUT_LINE(‘Department number: ‘||my_dname_table(int).deptno);
12 DBMS_OUTPUT.PUT_LINE(‘Department name: ‘|| my_dname_table(int).dname);
13 END LOOP;
14 END;
例12:按一维数组使用记录表
1 DECLARE
2 —定义记录表数据类型
3 TYPE reg_table_type IS TABLE OF varchar2(25)
4 INDEX BY BINARY_INTEGER;
5 —声明记录表数据类型的变量
6 v_reg_table REG_TABLE_TYPE;
7 BEGIN
8 v_reg_table(1) := ‘Europe’;
9 v_reg_table(2) := ‘Americas’;
10 v_reg_table(3) := ‘Asia’;
11 v_reg_table(4) := ‘Middle East and Africa’;
12 v_reg_table(5) := ‘NULL’;
13 DBMS_OUTPUT.PUT_LINE(‘地区名称:’||v_reg_table (1)||’、’
14 ||v_reg_table (2)||’、’
15 ||v_reg_table (3)||’、’
16 ||v_reg_table (4));
17 DBMS_OUTPUT.PUT_LINE(‘第5个成员的值:’||v_reg_table(5));
18 END;
例13:按二维数组使用记录表
1 DECLARE
2 —定义记录表数据类型
3 TYPE emp_table_type IS TABLE OF employees%ROWTYPE
4 INDEX BY BINARY_INTEGER;
5 —声明记录表数据类型的变量
6 v_emp_table EMP_TABLE_TYPE;
7 BEGIN
8 SELECT first_name, hire_date, job_id INTO
9 v_emp_table(1).first_name,v_emp_table(1).hire_date, v_emp_table(1).job_id
10 FROM employees WHERE employee_id = 177;
11 SELECT first_name, hire_date, job_id INTO
12 v_emp_table(2).first_name,v_emp_table(2).hire_date, v_emp_table(2).job_id
13 FROM employees WHERE employee_id = 178;
14 DBMS_OUTPUT.PUT_LINE(‘177雇员名称:’||v_emp_table(1).first_name
15 ||’ 雇佣日期:’||v_emp_table(1).hire_date
16 ||’ 岗位:’||v_emp_table(1).job_id);
17 DBMS_OUTPUT.PUT_LINE(‘178雇员名称:’||v_emp_table(2).first_name
18 ||’ 雇佣日期:’||v_emp_table(2).hire_date
19 ||’ 岗位:’||v_emp_table(2).job_id);
20 END;

2.5 运算符和表达式(数据定义)

2.5.1 关系运算符

2.5.2 一般运算符

2.5.3 逻辑运算符

2.6 变量赋值

在PL/SQL编程中,变量赋值是一个值得注意的地方,它的语法如下:
variable := expression ;
variable 是一个PL/SQL变量, expression 是一个PL/SQL 表达式.
2.6.1 字符及数字运算特点
空值加数字仍是空值:NULL + < 数字> = NULL
空值加(连接)字符,结果为字符:NULL || <字符串> = < 字符串>
2.6.2 BOOLEAN 赋值
布尔值只有TRUE, FALSE及 NULL 三个值。如:
1 DECLARE
2 bDone BOOLEAN;
3 BEGIN
4 bDone := FALSE;
5 WHILE NOT bDone LOOP
6 Null;
7 END LOOP;
8 END;
2.6.3 数据库赋值
数据库赋值是通过 SELECT语句来完成的,每次执行 SELECT语句就赋值一次,一般要求被赋值的变量与SELECT中的列名要一一对应。如:
例14:
1 DECLARE
2 emp_id emp.empno%TYPE :=7788;
3 emp_name emp.ename%TYPE;
4 wages emp.sal%TYPE;
5 BEGIN
6 SELECT ename, NVL(sal,0) + NVL(comm,0) INTO emp_name, wages
7 FROM emp WHERE empno = emp_id;
8 DBMS_OUTPUT.PUT_LINE(emp_name||’——‘||to_char(wages));
9 END;
提示:不能将SELECT语句中的列赋值给布尔变量。
2.6.4 可转换的类型赋值
1.CHAR 转换为 NUMBER:
使用 TO_NUMBER 函数来完成字符到数字的转换,如:
v_total := TO_NUMBER(‘100.0’) + sal;
2.NUMBER 转换为CHAR:
使用 TO_CHAR函数可以实现数字到字符的转换,如:
v_comm := TO_CHAR(‘123.45’) || ‘元’ ;
3.字符转换为日期:
使用 TO_DATE函数可以实现 字符到日期的转换,如:
v_date := TO_DATE(‘2001.07.03’,’yyyy.mm.dd’);
4.日期转换为字符
使用 TO_CHAR函数可以实现日期到字符的转换,如:
v_to_day := TO_CHAR(SYSDATE, ‘yyyy.mm.dd hh24:mi:ss’) ;

2.7 变量作用范围及可见性

在PL/SQL编程中,如果在变量的定义上没有做到统一的话,可能会隐藏一些危险的错误,这样的原因主要是变量的作用范围所致。变量的作用域是指变量的有效作用范围,与其它高级语言类似,PL/SQL的变量作用范围特点是:
1.变量的作用范围是在你所引用的程序单元(块、子程序、包)内。即从声明变量开始到该块的结束。
2.一个变量(标识)只能在你所引用的块内是可见的。
3.当一个变量超出了作用范围,PL/SQL引擎就释放用来存放该变量的空间(因为它可能不用了)。
4.在子块中重新定义该变量后,它的作用仅在该块内。
例15:

 DECLARE
    Emess char(80);
 BEGIN
    DECLARE
       V1 NUMBER(4);
    BEGIN
       SELECT empno INTO v1 FROM emp WHERE LOWER(job)='president';
       DBMS_OUTPUT.PUT_LINE(V1);
    EXCEPTION
       When TOO_MANY_ROWS THEN
          DBMS_OUTPUT.PUT_LINE ('More than one president');
    END;
    DECLARE
       V1 NUMBER(4);
    BEGIN
       SELECT empno INTO v1 FROM emp WHERE LOWER(job)='manager';
    EXCEPTION
       When TOO_MANY_ROWS THEN
           DBMS_OUTPUT.PUT_LINE ('More than one manager');
    END;
 EXCEPTION
    When others THEN
       Emess:=substr(SQLERRM,1,80);
       DBMS_OUTPUT.PUT_LINE(emess);
 END;

2.8 注释

在PL/SQL里,可以使用两种符号来写注释,即:使用双 ‘-‘ ( 减号) 加注释<br />    PL/SQL允许用 – 来写注释,它的作用范围是只能在一行有效。如:<br />    V_Sal  NUMBER(12,2); -- 人员的工资变量。<br />使用 /_   _/  来加一行或多行注释,如:<br />/***/<br />/_ 文件名: department_salary.sql      _/<br />/_ 作 者: EricHu                     _/<br />/_ 时 间: 2011-5-9                  _/<br />/***/<br />提示:被解释后存放在数据库中的 PL/SQL 程序,一般系统自动将程序头部的注释去掉。只有在 PROCEDURE 之后的注释才被保留;另外程序中的空行也自动被去掉。

3. PL/SQL流程控制语句

介绍PL/SQL的流程控制语句, 包括如下三类:
控制语句: IF 语句
循环语句: LOOP语句, EXIT语句
顺序语句: GOTO语句, NULL语句

3.1 条件语句

IF <布尔表达式> THEN
PL/SQL 和 SQL语句
END IF;
———————————-
IF <布尔表达式> THEN
PL/SQL 和 SQL语句
ELSE
其它语句
END IF;
———————————-
IF <布尔表达式> THEN
PL/SQL 和 SQL语句
ELSIF < 其它布尔表达式> THEN
其它语句
ELSIF < 其它布尔表达式> THEN
其它语句
ELSE
其它语句
END IF;
提示: ELSIF 不能写成 ELSEIF
例1:

 DECLARE
     v_empno  employees.employee_id%TYPE :=&empno;
     V_salary employees.salary%TYPE;
     V_comment VARCHAR2(35);
 BEGIN
 SELECT salary INTO v_salary FROM employees
 WHERE employee_id = v_empno;
 IF v_salary < 1500 THEN
        V_comment:= '太少了,加点吧~!';
    ELSIF v_salary <3000 THEN
       V_comment:= '多了点,少点吧~!';
 ELSE
       V_comment:= '没有薪水~!';
 END IF;
    DBMS_OUTPUT.PUT_LINE(V_comment);
    exception
 when no_data_found then
         DBMS_OUTPUT.PUT_LINE('没有数据~!');
 when others then
         DBMS_OUTPUT.PUT_LINE(sqlcode || '---' || sqlerrm);
 END;

例2:

 DECLARE
    v_first_name  VARCHAR2(20);
    v_salary NUMBER(7,2);
 BEGIN
 SELECT first_name, salary INTO v_first_name, v_salary FROM employees
 WHERE employee_id = &emp_id;
    DBMS_OUTPUT.PUT_LINE(v_first_name||'雇员的工资是'||v_salary);
 IF v_salary < 10000 THEN
       DBMS_OUTPUT.PUT_LINE('工资低于10000');
 ELSE
 IF 10000 <= v_salary AND v_salary < 20000 THEN
          DBMS_OUTPUT.PUT_LINE('工资在10000到20000之间');
 ELSE
          DBMS_OUTPUT.PUT_LINE('工资高于20000');
 END IF;
 END IF;
 END;

例3:

 DECLARE
    v_first_name  VARCHAR2(20);
    v_hire_date DATE;
    v_bonus NUMBER(6,2);
 BEGIN
 SELECT first_name, hire_date INTO v_first_name, v_hire_date FROM employees
 WHERE employee_id = &emp_id;
 IF v_hire_date > TO_DATE('01-1月-90') THEN
       v_bonus := 800;
    ELSIF v_hire_date > TO_DATE('01-1月-88') THEN
       v_bonus := 1600;
 ELSE
       v_bonus := 2400;
 END IF;
    DBMS_OUTPUT.PUT_LINE(v_first_name||'雇员的雇佣日期是'||v_hire_date
 ||'、奖金是'||v_bonus);
 END;

3.2 CASE 表达式

————-格式一————-
CASE 条件表达式
WHEN 条件表达式结果1 THEN
语句段1
WHEN 条件表达式结果2 THEN
语句段2
……
WHEN 条件表达式结果n THEN
语句段n
[ELSE 条件表达式结果]
END;
————-格式二————-
CASE
WHEN 条件表达式1 THEN
语句段1
WHEN 条件表达式2 THEN
语句段2
……
WHEN 条件表达式n THEN
语句段n
[ELSE 语句段]
END;
例4:

 DECLARE
   V_grade char(1) := UPPER('&p_grade');
   V_appraisal VARCHAR2(20);
 BEGIN
   V_appraisal :=
 CASE v_grade
 WHEN 'A' THEN 'Excellent'
 WHEN 'B' THEN 'Very Good'
 WHEN 'C' THEN 'Good'
 ELSE 'No such grade'
 END;
   DBMS_OUTPUT.PUT_LINE('Grade:'||v_grade||'  Appraisal: '|| v_appraisal);
 END;

例5:

 DECLARE
    v_first_name employees.first_name%TYPE;
    v_job_id employees.job_id%TYPE;
    v_salary employees.salary%TYPE;
    v_sal_raise NUMBER(3,2);
 BEGIN
 SELECT first_name,   job_id,   salary INTO
           v_first_name, v_job_id, v_salary
 FROM employees WHERE employee_id = &emp_id;
 CASE
 WHEN v_job_id = 'PU_CLERK' THEN
 IF v_salary < 3000 THEN v_sal_raise := .08;
 ELSE v_sal_raise := .07;
 END IF;
 WHEN v_job_id = 'SH_CLERK' THEN
 IF v_salary < 4000 THEN v_sal_raise := .06;
 ELSE v_sal_raise := .05;
 END IF;
 WHEN v_job_id = 'ST_CLERK' THEN
 IF v_salary < 3500 THEN v_sal_raise := .04;
 ELSE v_sal_raise := .03;
 END IF;
 ELSE
          DBMS_OUTPUT.PUT_LINE('该岗位不涨工资: '||v_job_id);
 END CASE;
    DBMS_OUTPUT.PUT_LINE(v_first_name||'的岗位是'||v_job_id
 ||'、的工资是'||v_salary
 ||'、工资涨幅是'||v_sal_raise);
 END;

3.3 循环

1. 简单循环

LOOP
要执行的语句;
EXIT WHEN <条件语句> —条件满足,退出循环语句
END LOOP;
例 6:
1 DECLARE
2 int NUMBER(2) :=0;
3 BEGIN
4 LOOP
5 int := int + 1;
6 DBMS_OUTPUT.PUT_LINE(‘int 的当前值为:’||int);
7 EXIT WHEN int =10;
8 END LOOP;
9 END;

2. WHILE 循环

WHILE <布尔表达式> LOOP
要执行的语句;
END LOOP;
例7.
1 DECLARE
2 x NUMBER :=1;
3 BEGIN
4 WHILE x<=10 LOOP
5 DBMS_OUTPUT.PUT_LINE(‘X的当前值为:’||x);
6 x:= x+1;
7 END LOOP;
8 END;

3. 数字式循环

[<<循环标签>>]
FOR 循环计数器 IN [ REVERSE ] 下限 .. 上限 LOOP
要执行的语句;
END LOOP [循环标签];
每循环一次,循环变量自动加1;使用关键字REVERSE,循环变量自动减1。跟在IN REVERSE 后面的数字必须是从小到大的顺序,而且必须是整数,不能是变量或表达式。可以使用EXIT 退出循环。
例8.
BEGIN
FOR int in 1..10 LOOP
DBMSOUTPUT.PUT_LINE(‘int 的当前值为: ‘||int);
END LOOP;
END;
例11 在While循环中嵌套loop循环:
1 /
求100至110之间的素数/
2 DECLARE
3 v_m NUMBER := 101;
4 v_i NUMBER;
5 v_n NUMBER := 0;
6 BEGIN
7 WHILE v_m < 110 LOOP
8 v_i := 2;
9 LOOP
10 IF mod(v_m, v_i) = 0 THEN
11 v_i := 0;
12 EXIT;
13 END IF;
14 v_i := v_i + 1;
15 EXIT WHEN v_i > v_m - 1;
16 END LOOP;
17 IF v_i > 0 THEN
18 v_n := v_n + 1;
19 DBMS_OUTPUT.PUT_LINE(‘第’|| v_n || ‘个素数是’ || v_m);
20 END IF;
21 v_m := v_m + 2;
22 END LOOP;
23 END;
3.4 标号和GOTO
PL/SQL中GOTO语句是无条件跳转到指定的标号去的意思。语法如下:
GOTO label;
……
<> /
标号是用<< >>括起来的标识符 _/
注意,在以下地方使用是不合法的,编译时会出错误。
u 跳转到非执行语句前面。
u 跳转到子块中。
u 跳转到循环语句中。
u 跳转到条件语句中。
u 从异常处理部分跳转到执行。
u 从条件语句的一部分跳转到另一部分。
例12:
1 DECLARE
2 V_counter NUMBER := 1;
3 BEGIN
4 LOOP
5 DBMS_OUTPUT.PUT_LINE(‘V_counter的当前值为:’||V_counter);
6 V_counter := v_counter + 1;
7 IF v_counter > 10 THEN
8 GOTO labelOffLOOP;
9 END IF;
10 END LOOP;
11 <>
12 DBMS_OUTPUT.PUT_LINE(‘V_counter的当前值为:’||V_counter);
13 END;
3.5 NULL 语句
在PL/SQL 程序中,NULL语句是一个可执行语句,可以用 null 语句来说明“不用做任何事情”的意思,相当于一个占位符或不执行任何操作的空语句,可以使某些语句变得有意义,提高程序的可读性,保证其他语句结构的完整性和正确性。如:
例14:
1 DECLARE
2 …
3 BEGIN
4 …
5 IF v_num IS NULL THEN
6 GOTO labelPrint;
7 END IF;
8 …
9 <>
10 NULL; —不需要处理任何数据。
11 END;

循环使用建议

1.简单LOOP循环:当需要确保循环体至少被执行一次时,可以使用LOOP END LOOP循环,在循环体中必须使用EXIT WHEN或EXIT语句结束循环,避免出现死循环。
2.WHILE-LOOP循环:当需要在循环开始前判断循环条件时,这种循环体可能使得循环体一次都不执行,在循环体内更改WHILE循环判断条件,防止陷入死循环。
3.FOR循环:如果循环次数开始就知道,在FOR循环中尽量避免出现EXIT和EXIT WHEN。
建议:不要在FOR和WHILE循环中使用EXIT和EXIT WHILE子句,在循环内避免使用RETURN和GOTO语句。