Oracle学习(十一):PL/SQL

1.知识点:可以对照下面的录屏进行阅读

PL/SQL程序结构
declare
      说明部分    (变量说明,光标申明,例外说明 〕
begin
      语句序列   (DML语句〕… 
exception
      例外处理语句   
End;
/
------------------------------------------------------------------
--第一个PL/SQL程序:HelloWorld
set serveroutput on	--如果要在屏幕上输出信息,需要将serveroutput开关打开
declare
  --变量的说明
begin
   --程序体
   --程序包package
   dbms_output.put_line('Hello World');
end;
/
------------------------------------------------------------------

--引用型变量: 查询并打印7839的姓名和薪水

set serveroutput on

declare 
  --定义变量,
  pename emp.ename%type;		--将pename的类型设置为emp表中ename的类型
  psal   emp.sal%type;	--将psal的类型设置为emp表中sal的类型

begin

  --查询: 
  --ename,sal into pename,psal:将查询的ename,sal分别赋值给pename,psal按顺序赋值
  select ename,sal into pename,psal from emp where empno=7839;
  
  --打印
  dbms_output.put_line(pename||'的薪水是'||psal);

end;
/
------------------------------------------------------------------

--记录型变量 查询并打印7839的姓名和薪水

set serveroutput on

declare 
  --定义变量,代表一行
  emp_rec emp%rowtype;
begin
  --得到一行,赋值给emp_rec
  select * into emp_rec from emp where empno=7839;

  dbms_output.put_line(emp_rec.ename||'的薪水是'||emp_rec.sal);

end;
/
------------------------------------------------------------------
--if语句:判断用户输入的数字

set serveroutput on

/*
1.提示信息
2. 接收键盘输入
num 是一个地址值

SQL优化: num绑定变量(尽量使用绑定变量)
select * from emp where deptno=10;  --> 执行计划
select * from emp where deptno=20;  --> 执行计划
-->
select * from emp where deptno=#

*/
accept num prompt '请输入一个数字';

declare
  --变量保存输入的数字
  pnum number  := #
begin
  --判断
  if pnum = 0 then
    dbms_output.put_line('您输入的是0');
  elsif pnum = 1 then
    dbms_output.put_line('您输入的是1');
  elsif pnum = 2 then
    dbms_output.put_line('您输入的是2');
  else
    dbms_output.put_line('其他数字');
  end if;    
end;
/
------------------------------------------------------------------
--循环: 打印1~10,有3中方法,见下图

set serveroutput on

declare
  pnum number := 1;	--给pnum赋值1,pnum类型为number
begin
  loop
    --退出:成立退出,不成立循环
    exit when pnum > 10;
    
    --隐式转换,number转varchar2
    dbms_output.put_line(pnum);
    
    pnum := pnum + 1;

  end loop;
end;
/
------------------------------------------------------------------

光标(Cursor)==ResultSet

说明光标语法:
CURSOR  光标名  [ (参数名  数据类型[,参数名 数据类型]...)]
      IS  SELECT   语句;
用于存储一个查询返回的多行数据
例如:
cursor c1 is select ename from emp;
打开光标:      open c1;    (打开光标执行查询)
取一行光标的值:fetch c1 into pjob; (取一行到变量中)
关闭光标:      close  c1;(关闭游标释放资源)
注意: 上面的pjob必须与emp表中的job列类型一致:
定义:pjob emp.empjob%type;
------------------------------------------------------------------
--光标: 使用游标查询员工姓名和工资,并打印

/*
光标的3个属性:
%isopen 是否被打开:打开true,关闭false;
%rowcount 行数
%notfound 是否有值:没有值true,有值false;
*/

set serveroutput on

declare
  --光标
  cursor cemp is select ename,sal from emp;
  pename emp.ename%type;		--将用来接收的变量类型设置为和查询的类型相同
  psal   emp.sal%type;	
begin
  open cemp;		--打开光标执行查询
  loop
    --从集合中取值
    fetch cemp into pename,psal;
    --光标中没有值时退出
    exit when cemp%notfound;
    
    dbms_output.put_line(pename||'的薪水是'||psal);

  end loop;
  close cemp;
end;
/
------------------------------------------------------------------
--光标练习:给员工涨工资  总裁1000 经理800 其他400
--光标同时能打开的最大数量默认为300个
/*
SQL> show parameters cursor

NAME                                 TYPE        VALUE
------------------------------------ ----------- --------
cursor_sharing                       string      EXACT
cursor_space_for_time                boolean     FALSE
open_cursors                         integer     300
session_cached_cursors               integer     20

*/
set serveroutput on
declare
  --光标代表员工
  cursor cemp is select empno,job from emp;
  pempno emp.empno%type;
  pjob   emp.job%type;
begin
  open cemp;
  loop
    fetch cemp into pempno,pjob;
    exit when cemp%notfound;
    
    --判断
    if pjob = 'PRESIDENT' then update emp set sal=sal+1000 where empno=pempno;	--总裁涨1000
      elsif pjob = 'MANAGER' then update emp set sal=sal+800 where empno=pempno; --经理涨800
      else update emp set sal=sal+400 where empno=pempno;	--其他涨400
    end if;
  end loop;
  close cemp;
  
  --提交: 隔离级别
  commit;
  
  dbms_output.put_line('完成');
end;
/
------------------------------------------------------------------
--带参数的光标:查询某个部门的员工姓名 

set serveroutput on

declare
  --定义一个带参数的光标
  cursor cemp(pdno number) is  select ename from emp where deptno=pdno;
  pename emp.ename%type;
begin
  open cemp(20);	--打开光标,并传递参数
  loop
    fetch cemp into pename;
    exit when cemp%notfound;
    
    dbms_output.put_line(pename);


  end loop;
  close cemp;
end;
/
------------------------------------------------------------------
Oracle的异常处理
1.系统定义例外
No_data_found    (没有找到数据)
Too_many_rows          (select …into语句匹配多个行) 
Zero_Divide   ( 被零除)
Value_error     (算术或转换错误)
Timeout_on_resource      (在等待资源时发生超时)

--举例 Zero_Divide   ( 被零除)

set serveroutput on
declare
  pnum number;
begin

  pnum := 1/0;

exception
  when Zero_Divide then dbms_output.put_line('1: 0不能做被除数');
                        dbms_output.put_line('2: 0不能做被除数');
  when Value_error then dbms_output.put_line('算术错');
  when others then dbms_output.put_line('其他例外');
end;
/

用户自己定义的例外
--自定义例外: 查询50号部门的员工姓名

set serveroutput on

declare
  cursor cemp is select ename from emp where deptno=50;
  pename emp.ename%type;
  
  --自定义例外
  no_emp_found exception;
begin
  open cemp;
  --取一个员工
  fetch cemp into pename;
  if cemp%notfound then 
    raise no_emp_found;
  end if;

/*
  if cemp%isopen then 
    close no_emp_found;
  end if;
*/
  close cemp;
  
exception 
  when no_emp_found then dbms_output.put_line('没有找到员工');
  when others then dbms_output.put_line('其他例外');
  
end;
/



图:循环的3种写法,文中采用右上角的方法。

2.在Sqlplus下实际执行的结果录屏

SQL> --第一个PL/SQL程序:HelloWorld
SQL> set serveroutput on	--如果要在屏幕上输出信息,需要将serveroutput开关打开
SQL> declare
  2    --变量的说明
  3  begin
  4     --程序体
  5     --程序包package
  6     dbms_output.put_line('Hello World');
  7  end;
  8  /
 
Hello World
 
PL/SQL procedure successfully completed
 
SQL> 
SQL> --引用型变量: 查询并打印7839的姓名和薪水
SQL> set serveroutput on
SQL> declare
  2    --定义变量,
  3    pename emp.ename%type;		--将pename的类型设置为emp表中ename的类型
  4    psal   emp.sal%type;	--将psal的类型设置为emp表中sal的类型
  5  
  6  begin
  7  
  8    --查询:
  9    --ename,sal into pename,psal:将查询的ename,sal分别赋值给pename,psal按顺序赋值
 10    select ename,sal into pename,psal from emp where empno=7839;
 11  
 12    --打印
 13    dbms_output.put_line(pename||'的薪水是'||psal);
 14  
 15  end;
 16  /
 
KING的薪水是5000
 
PL/SQL procedure successfully completed
 
SQL> 
SQL> -记录型变量 查询并打印7839的姓名和薪水
SQL> set serveroutput on
SQL> declare
  2    --定义变量,代表一行
  3    emp_rec emp%rowtype;
  4  begin
  5    --得到一行,赋值给emp_rec
  6    select * into emp_rec from emp where empno=7839;
  7  
  8    dbms_output.put_line(emp_rec.ename||'的薪水是'||emp_rec.sal);
  9  
 10  end;
 11  /
 
KING的薪水是5000
 
PL/SQL procedure successfully completed
 
SQL> 
SQL> --if语句:判断用户输入的数字
SQL> set serveroutput on
SQL> /*
  2  1.提示信息
  3  2. 接收键盘输入
  4  num 是一个地址值
  5  
  6  SQL优化: num绑定变量(尽量使用绑定变量)
  7  select * from emp where deptno=10;  --> 执行计划
  8  select * from emp where deptno=20;  --> 执行计划
  9  -->
 10  select * from emp where deptno=#
 11  
 12  */
 13  accept num prompt '请输入一个数字';
SQL> declare
  2    --变量保存输入的数字
  3    pnum number  := #
  4  begin
  5    --判断
  6    if pnum = 0 then
  7      dbms_output.put_line('您输入的是0');
  8    elsif pnum = 1 then
  9      dbms_output.put_line('您输入的是1');
 10    elsif pnum = 2 then
 11      dbms_output.put_line('您输入的是2');
 12    else
 13      dbms_output.put_line('其他数字');
 14    end if;
 15  end;
 16  /
 
您输入的是2
 
PL/SQL procedure successfully completed
 
SQL> 
SQL> --循环: 打印1~10
SQL> set serveroutput on
SQL> declare
  2    pnum number := 1;	--给pnum赋值1,pnum类型为number
  3  begin
  4    loop
  5      --退出:成立退出,不成立循环
  6      exit when pnum > 10;
  7  
  8      --隐式转换,number转varchar2
  9      dbms_output.put_line(pnum);
 10  
 11      pnum := pnum + 1;
 12  
 13    end loop;
 14  end;
 15  /
 
1
2
3
4
5
6
7
8
9
10
 
PL/SQL procedure successfully completed
 
SQL> 
SQL> -光标: 使用游标查询员工姓名和工资,并打印
SQL> /*
  2  光标的3个属性:
  3  %isopen 是否被打开:打开true,关闭false;
  4  %rowcount 行数
  5  %notfound 是否有值:没有值true,有值false;
  6  */
SQL> set serveroutput on
SQL> declare
  2    --光标
  3    cursor cemp is select ename,sal from emp;
  4    pename emp.ename%type;		--将用来接收的变量类型设置为和查询的类型相同
  5    psal   emp.sal%type;
  6  begin
  7    open cemp;		--打开光标执行查询
  8    loop
  9      --从集合中取值
 10      fetch cemp into pename,psal;
 11      --光标中没有值时退出
 12      exit when cemp%notfound;
 13  
 14      dbms_output.put_line(pename||'的薪水是'||psal);
 15  
 16    end loop;
 17    close cemp;
 18  end;
 19  /
 
SMITH的薪水是800
ALLEN的薪水是1600
WARD的薪水是1250
JONES的薪水是2975
MARTIN的薪水是1250
BLAKE的薪水是2850
CLARK的薪水是2450
SCOTT的薪水是3000
KING的薪水是5000
TURNER的薪水是1500
ADAMS的薪水是1100
JAMES的薪水是950
FORD的薪水是3000
MILLER的薪水是1300
 
PL/SQL procedure successfully completed
 
SQL> --光标练习:给员工涨工资  总裁1000 经理800 其他400
SQL> --先查询没涨之前的
SQL> select * from emp;
 
EMPNO ENAME      JOB         MGR HIREDATE          SAL      COMM DEPTNO
----- ---------- --------- ----- ----------- --------- --------- ------
 7369 SMITH      CLERK      7902 1980/12/17     800.00               20
 7499 ALLEN      SALESMAN   7698 1981/2/20     1600.00    300.00     30
 7521 WARD       SALESMAN   7698 1981/2/22     1250.00    500.00     30
 7566 JONES      MANAGER    7839 1981/4/2      2975.00               20
 7654 MARTIN     SALESMAN   7698 1981/9/28     1250.00   1400.00     30
 7698 BLAKE      MANAGER    7839 1981/5/1      2850.00               30
 7782 CLARK      MANAGER    7839 1981/6/9      2450.00               10
 7788 SCOTT      ANALYST    7566 1987/4/19     3000.00               20
 7839 KING       PRESIDENT       1981/11/17    5000.00               10
 7844 TURNER     SALESMAN   7698 1981/9/8      1500.00      0.00     30
 7876 ADAMS      CLERK      7788 1987/5/23     1100.00               20
 7900 JAMES      CLERK      7698 1981/12/3      950.00               30
 7902 FORD       ANALYST    7566 1981/12/3     3000.00               20
 7934 MILLER     CLERK      7782 1982/1/23     1300.00               10
 
14 rows selected
 
SQL> --执行PLSQL语句
SQL> 
SQL> set serveroutput on
SQL> declare
  2    --光标代表员工
  3    cursor cemp is select empno,job from emp;
  4    pempno emp.empno%type;
  5    pjob   emp.job%type;
  6  begin
  7    open cemp;
  8    loop
  9      fetch cemp into pempno,pjob;
 10      exit when cemp%notfound;
 11  
 12      --判断
 13      if pjob = 'PRESIDENT' then update emp set sal=sal+1000 where empno=pempno;	--总裁涨1000
 14        elsif pjob = 'MANAGER' then update emp set sal=sal+800 where empno=pempno; --经理涨800
 15        else update emp set sal=sal+400 where empno=pempno;	--其他涨400
 16      end if;
 17    end loop;
 18    close cemp;
 19  
 20    --提交: 隔离级别
 21    commit;
 22  
 23    dbms_output.put_line('完成');
 24  end;
 25  /
 
完成
 
PL/SQL procedure successfully completed
 
SQL> --查询执行PLSQL语句后的薪水
SQL> select * from emp;
 
EMPNO ENAME      JOB         MGR HIREDATE          SAL      COMM DEPTNO
----- ---------- --------- ----- ----------- --------- --------- ------
 7369 SMITH      CLERK      7902 1980/12/17    1200.00               20
 7499 ALLEN      SALESMAN   7698 1981/2/20     2000.00    300.00     30
 7521 WARD       SALESMAN   7698 1981/2/22     1650.00    500.00     30
 7566 JONES      MANAGER    7839 1981/4/2      3775.00               20
 7654 MARTIN     SALESMAN   7698 1981/9/28     1650.00   1400.00     30
 7698 BLAKE      MANAGER    7839 1981/5/1      3650.00               30
 7782 CLARK      MANAGER    7839 1981/6/9      3250.00               10
 7788 SCOTT      ANALYST    7566 1987/4/19     3400.00               20
 7839 KING       PRESIDENT       1981/11/17    6000.00               10
 7844 TURNER     SALESMAN   7698 1981/9/8      1900.00      0.00     30
 7876 ADAMS      CLERK      7788 1987/5/23     1500.00               20
 7900 JAMES      CLERK      7698 1981/12/3     1350.00               30
 7902 FORD       ANALYST    7566 1981/12/3     3400.00               20
 7934 MILLER     CLERK      7782 1982/1/23     1700.00               10
 
14 rows selected
 
SQL> --对比涨之前,可以看出涨工资成功
SQL> 
SQL> --带参数的光标:查询某个部门的员工姓名
SQL> set serveroutput on
SQL> declare
  2    --定义一个带参数的光标
  3    cursor cemp(pdno number) is  select ename from emp where deptno=pdno;
  4    pename emp.ename%type;
  5  begin
  6    open cemp(20);	--打开光标,并传递参数
  7    loop
  8      fetch cemp into pename;
  9      exit when cemp%notfound;
 10  
 11      dbms_output.put_line(pename);
 12  
 13  
 14    end loop;
 15    close cemp;
 16  end;
 17  /
 
SMITH
JONES
SCOTT
ADAMS
FORD
 
PL/SQL procedure successfully completed
 
SQL> 
SQL> --举例 Zero_Divide   ( 被零除)
SQL> set serveroutput on
SQL> declare
  2    pnum number;
  3  begin
  4  
  5    pnum := 1/0;
  6  
  7  exception
  8    when Zero_Divide then dbms_output.put_line('1: 0不能做被除数');
  9                          dbms_output.put_line('2: 0不能做被除数');
 10    when Value_error then dbms_output.put_line('算术错');
 11    when others then dbms_output.put_line('其他例外');
 12  end;
 13  /
 
1: 0不能做被除数
2: 0不能做被除数
 
PL/SQL procedure successfully completed
 
SQL> 
SQL> --自定义例外: 查询50号部门的员工姓名
SQL> set serveroutput on
SQL> declare
  2    cursor cemp is select ename from emp where deptno=50;
  3    pename emp.ename%type;
  4  
  5    --自定义例外
  6    no_emp_found exception;
  7  begin
  8    open cemp;
  9    --取一个员工
 10    fetch cemp into pename;
 11    if cemp%notfound then
 12      raise no_emp_found;
 13    end if;
 14  
 15    close cemp;
 16  
 17  exception
 18    when no_emp_found then dbms_output.put_line('没有找到员工');
 19    when others then dbms_output.put_line('其他例外');
 20  
 21  end;
 22  /
 
没有找到员工
 
PL/SQL procedure successfully completed
 
SQL> spool off
Stopped spooling to c:\PLSQL.txt


3.例子

/*
实例1:统计每年入职的员工个数。

用到的SQL:
select to_char(hiredate,'yyyy') from emp;
*/
set serveroutput on
declare
  cursor cemp is select to_char(hiredate,'yyyy') from emp;
  phiredate varchar2(4);
  
  --计数器
  count80 number := 0;
  count81 number := 0;
  count82 number := 0;
  count87 number := 0;
begin
  open cemp;
  loop
    --取一个员工
    fetch cemp into phiredate;
    exit when cemp%notfound;
    
    --判断
    if phiredate = '1980' then count80:=count80+1;
      elsif phiredate = '1981' then count81:=count81+1;
      elsif phiredate = '1982' then count82:=count82+1;
      else count87 := count87+1;
    end if;      

  end loop;
  close cemp;
  
  --输出
  dbms_output.put_line('total:'||(count80+count81+count82+count87));
  dbms_output.put_line('1980:'|| count80);
  dbms_output.put_line('1981:'|| count81);
  dbms_output.put_line('1982:'|| count82);
  dbms_output.put_line('1987:'|| count87);
end;
/
------------------------------------------------------------------

/*
实例2:为员工长工资,从最低工资调起每人长10%,但工资总额不能超过5万元,
请计算长工资的人数和长工资后的工资总额,并输出输出长工资人数及工资总额。

可能的SQL:
员工:  select empno,sal from emp order by sal;
长工资后的工资总额:1. 对sal进行累加: 新的工资总额=旧的工资 + sal*0.1;
                2. sum(sal): 查询数据库
练习: 工资不能超过5w
*/
set serveroutput on
declare
  --员工
  cursor cemp is select empno,sal from emp order by sal;
  pempno emp.empno%type;
  psal   emp.sal%type;
  
  --长工资的人数
  countEmp number := 0;
  --工资总额
  salTotal number;
begin

   --涨前工资总额
  select sum(sal) into salTotal from emp;

  open cemp;
  loop
    --工资总额>5w
    exit when salTotal > 50000;
    --取一个员工
    fetch cemp into pempno,psal;
    exit when cemp%notfound;
    
    --涨工资
    update emp set sal=sal*1.1 where empno=pempno;
    --人数
    countEmp := countEmp +1;
    --工资总额
    salTotal := salTotal + psal * 0.1;
  end loop;
  close cemp;
  
  commit;

  --输出
  dbms_output.put_line('长工资的人数:'|| countEmp);
  dbms_output.put_line('工资总额:'|| salTotal);

end;
/
------------------------------------------------------------------
/*
实例3:用PL/SQL语言编写一程序,实现按部门分段(6000以上、(6000,3000)、3000元以下)
统计各工资段的职工人数、以及各部门的工资总额(工资总额中不包括奖金)

SQL语句:
部门:  select deptno from dept;
员工的工资: select sal from emp where deptno=???
工资总额:  select sum(sal) from emp where deptno=???  
*/
set serveroutput on
declare
  --部门 
  cursor cdept is select deptno from dept;
  pdno dept.deptno%type;
  
  --部门中的员工
  cursor cemp(dno number) is select sal from emp where deptno=dno;
  psal emp.sal%type;
  
  --各个段的人数
  count1 number;count2 number;count3 number;
  --部门的工资总额
  salTotal number;
begin
  open cdept;
  loop
    --取部门
    fetch cdept into pdno;
    exit when cdept%notfound;
    
    --初始化
    count1 :=0;count2:=0;count3:=0;
    select sum(sal) into salTotal  from emp where deptno=pdno;
    
    --取部门中的员工
    open cemp(pdno);
    loop
      fetch cemp into psal;
      exit when cemp%notfound;
      
      --判断
      if psal<3000 then count1:=count1+1;
        elsif psal>=3000 and psal<6000 then count2:=count2+1;
        else count3:=count3+1;
      end if;        
    end loop;
    close cemp;
    
    --保存当前部门
    insert into msg1 values(pdno,count1,count2,count3,nvl(salTotal,0));
    
  end loop;
  close cdept;
  
  commit;
  dbms_output.put_line('完成');



end;
/


你可能感兴趣的:(java,oracle,sql,数据库,plsql)