oracle--PLSQL

PLSQL

一、基本概念

最简单的pl/sql程序--Hello World

要在屏幕上输出信息,需要将serveroutput开关打开:
set serveroutput on

declare
  --说明部分
begin
  --程序
  dbms_output.put_line('Hello World');
end;
/

查找oracle提供的程序包:

在对应数据文档中的index.htm中:
Books-->PL/-->
PL/SQL Packages and Types Reference:帮助文档
PL/SQL User's Guide and Reference:语法

也可以在命令行中直接查看程序包的结构:
desc+程序包的名字:
desc dbms_output

1.什么是PL/SQL

PL/SQL(Procedure Language/SQL)
PL/SQL是oracle对sql语言的过程化扩展
指在sql领命语言中增加了过程处理语句(如分支、循环等),是SQL语言具有过程处理能力。

2.PL/SQL程序结构

declare 
   说明部分(变量说明,光标申明,例外说明)
begin
   语句序列(DML语句)...
exception
   例外处理语句
end;

变量和常量说明:

说明变量:char,varchar2,date,number,boolean,long

说明变量名、数据类型和长度后用分号结束说明语句:
var1 char(15);  
married boolean :=true; 
psal number(7,2);

引用型变量,即my_name的类型与emp表中ename列的类型一样
my_name emp.ename%type;

记录型变量
emp_rec emp%rowtype;

记录变量分量的引用:
emp_rec.ename:='adams';

常量:在中间加上constant:
var1 constant char(15);

例:

引用型变量:
--查询7839的姓名和薪水
set serveroutput on

declare
  --定义变量保存姓名和薪水
  --pename varchar2(20);
  --psal   number;
  pename emp.ename%type;
  psal   emp.sal%type;
begin
  --得到姓名和薪水
  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
  select * into emp_rec from emp where empno=7839;
  dbms_output.put_line(emp_rec.ename||'的薪水是'||emp_rec.sal);
end;

注意:

赋值的方式:
1.使用冒号等号:=
2.使用into,into前后变量的顺序要保持一致

if语句:

1.if 条件 then 语句1;
  语句2;
  end if;

2.if 条件 then 语句序列1;
  else 语句序列2;
  end if;

3.if 条件 then 语句;
  elseif 语句 then 语句;
  else 语句;
  end if;

例:

--判断用户从键盘输入的数字

set serveroutput on

--接收键盘输入
--num: 地址值,在该地址上 保存了输入的值
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;

循环语句:

当条件成立,执行循环;不成立,退出循环
while total<=25000
loop
...
total:=total+salary;
end loop;

当条件成立,退出循环;不成立,执行循环
loop
exit[when 条件];
...
end loop;

连续范围的时候才可以用..省略
for i in 1..3
loop
语句序列;
end loop;

例:

--打印1~10
set serveroutput on

declare
  pnum number := 1;
begin
  loop
    --退出条件
    exit when pnum > 10;
    
    dbms_output.put_line(pnum);
    --加一
    pnum := pnum + 1;
  end loop;
end;

二、光标

语法:

cursor 光标名[(参数名 数据类型[,参数名 数据类型]...)]
  is select 语句;

用户存储一个查询返回的多行数据

cursor c1 is select ename from emp;

打开光标: open c1;(打开光标执行查询)(光标的初始位置直接是指向了第一条记录的)
取一行光标的值:fetch c1 into pename;(取一行到变量中)
关闭光标:close c1;(关闭游标释放资源)
注意:
上面的pename必须与emp表中的ename列类型一致:
定义:pename emp.ename%type;

属性:

1.光标的属性:
  %isopen     %rowcount (影响的行数)
  %found      %notfound (是否有取到记录)

2.默认,一个会话中只能打开300个光标
SQL> show parameter cursor
NAME                                 TYPE                             VALUE
------------------------------------ -------------------------------- -----------
cursor_sharing                       string                           FORCE
cursor_space_for_time                boolean                          FALSE
open_cursors                         integer                          300
session_cached_cursors               integer                          20

注意:
要管理员身份才能修改: 
SQL> show user
USER is "SCOTT"
SQL> conn sys/password@10.10.10.128/orcl as sysdba
Connected.
SQL> show user
USER is "SYS"
alter system set open_cursors=400;

3.(思考):cursor_sharing 什么作用?---> 性能优化
     三个取值: EXACT(默认), FORCE, SIMILAR    

例:

--查询并打印员工的姓名和薪水
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 没有取到记录;
    exit when cemp%notfound;

    dbms_output.put_line(pename||'的薪水是'||psal);

  end loop;
  
  --关闭光标
  close cemp;
end;

--涨工资,总裁1000 经理800 其他400
set serveroutput on

declare 
  --alter table "SCOTT"."EMP" rename column "JOB" to empjob
  cursor cemp is select empno,empjob from emp;
  pempno emp.empno%type;
  pjob   emp.empjob%type;
begin
  rollback;

  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;
      elsif pjob = 'MANAGER' then update emp set sal=sal+800 where empno=pempno;
      else update emp set sal=sal+400 where empno=pempno;
    end if;
    
  end loop;
  close cemp;
  
  --why?  ---> ACID
  commit;
  
  dbms_output.put_line('完成');
end;

带参数的光标:

与不带参数的光标的区别是在定义是要带形参和打开时赋值:

--查询某个部门的员工姓名
set serveroutput on

declare
  cursor cemp(dno number) is select ename from emp where deptno=dno;
  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的异常处理:

系统定义例外:
no_data_found(没有找到数据)
too_many_rows(select...into语句匹配多个行)
zero_divide(被零除)
value_error(算术或转换错误)
timeout_on_resource(在等待资源时发生超时)

用户定义的例外

语法:

系统定义:
declare
    说明部分(变量说明,光标申明,例外说明)
begin
    语句序列(DML语句)...
exception
    例外处理语句
end;

自定义:
declare
my_job char(10);
v_sal emp.sal%type;
no_data exception;
cursor c1 is select
distinct job from emp
order by job;   
begin
open c1;
fetch c1 into v_job;
if c1%notfound then raise no_data;
end if;
...
exception
when no_data then insert into emp 
values('fetch语句没有获得数据或数据已经处理完');
end;

在declare节中定义例外:
out_of exception;
在可行语句中引起例外:
raise out_of;
在exception节处理例外:
when out_of then...

例:

系统例外
--被0除
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;

  --pmon: process monitor
  close cemp;
  
exception
  when no_emp_found then dbms_output.put_line('没有找到员工');
  when others then dbms_output.put_line('其他例外'); 
end;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,761评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,953评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,998评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,248评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,130评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,145评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,550评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,236评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,510评论 1 291
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,601评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,376评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,247评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,613评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,911评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,191评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,532评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,739评论 2 335

推荐阅读更多精彩内容