oracle学习篇(四)

news2025/1/16 18:02:51

oracle学习篇(四)

1 PL/SQL异常处理

1.1 预定义异常

1.1.1 内容

oracle里面已经存在的异常
如果是自定义异常,一般写的编号是20000-20999之间

在这里插入图片描述

1.1.2 处理异常语法

exception
     when 异常类型1 then
         输出异常类型信息1;
     when 异常信息2 then
         输出异常类型信息2;
     --以上都不符合的时候,就去执行后续的代码   
     when others then   
        输出以上异常类型都不满足时的信息;

1.1.3 处理异常代码

declare 
   vid number:=8888;
   vrow emp%rowtype;
begin
  SELECT * into vrow FROM emp;
  --dbms_output.put_line(1/0);
  --异常 大于2000开始 弹出来的好一点
  exception
     when ZERO_DIVIDE then
        dbms_output.put_line('除数不能为0');
     when NO_DATA_FOUND then
         dbms_output.put_line('into时某一查询到任何数据');
     --否则的意思    
     when others then   
       dbms_output.put_line('发生未知异常');   
end;

1.1.4 运行截图

在这里插入图片描述

1.2 自定义异常

1.2.1 以弹窗的方式进行抛出

a 语法
raise_application_error(自定义异常编号,'工资太低异常');
-- 自定义异常编号范围通常是在[-20001,-20999]
b 示例代码
declare
 vsal number;
begin
  SELECT sal into vsal from emp where empno=7369;
  if vsal<3000 then
      --弹窗形式 无法捕获
    raise_application_error(-20001,'工资太低异常');
  else
    dbms_output.put_line('工资是:'||vsal);  
  end if;  
end;
c 示例代码运行截图

在这里插入图片描述

1.2.2 抛出异常,可以通过exception去进行捕获的

a 语法
declare
 自定义异常名 exception;
begin
  raise 自定义异常名;  
  exception 
    when 自定义异常名 then
      异常处理语句; 
en
b 示例代码
-- 抛出异常的方式 可以通过exception去捕获,myex是异常对象
declare
 vsal number;
 myex exception;
begin
  SELECT sal into vsal from emp where empno=7369;
  if vsal<3000 then
      --抛出异常 捕获的方式去写
    raise myex;  
  else
    dbms_output.put_line('工资是:'||vsal);  
  end if;
  -- 捕获产生的异常
  exception 
    when myex then
    dbms_output.put_line('工资低了');      
end;
c 示例代码运行截图

在这里插入图片描述

2 游标

2.1 游标初识

游标cursor 等同java中的ResultSet结果集对象 存储的就是一个表,存储的是多行多列的数据,
解决了之前%rowtype仅能存储一行的尴尬局面
   能够存储多行多列的数据
   1.隐式游标
      oracle自带的游标,是不需要去声明的
      sql%rowcount 得到最近DML的受影响行数
   2  显式游标
    1 静态游标 cursor 游标的结果集在定义是就确定了
   2  ref游标 动态游标 sys_refcursor 能够在开启时动态赋值结果集内容
   sys可以作为参数去传递的

2.2 显示游标

2.2.1 静态游标

a 语法
declare
   --静态时 定义时确定好了游标的值
   cursor 游标值 is select * from dept;
b 示例代码
declare
   --静态时 定义时确定好了游标的值
   cursor vdept is select * from dept;
   vrow dept%rowtype;-- 用于接收游标的每一行数据
begin
     -- 开启游标 
     open vdept;
     --遍历游标
     loop
        --每次提取一行数据
       fetch vdept into vrow; 
       exit when vdept%notfound;--当游标没有数据时退出
       dbms_output.put_line('编号:'||vrow.deptno||'部门名'||vrow.dname);  
     end loop;  
end;
c 示例代码运行截图

在这里插入图片描述

2.2.2 动态游标

a 语法
-- 动态游标三步骤:①通过sys_refcursor,②  open cemp for去开启游标③ fetch cemp into 取出游标中的值
declare
   cemp sys_refcursor;--定义游标
   --自定义变量
   a varchar2(20);
   b varchar2(20);
   -- 自定义变量类型
begin
     -- 开启游标 用for去开启
     -- open cemp for select * from emp where deptno=vdeptno; --关闭是close
     open 游标名 for select语句
end;     
b 示例代码
declare
   -- 游标的值是根据输入的内容来决定的
   vdeptno number:=&请输入部门编号;
   cemp sys_refcursor;--定义游标
   vrow emp%rowtype;
   --自定义变量
   a varchar2(20);
   b varchar2(20);
   -- 自定义变量类型
begin
     -- 开启游标 用for去开启
     -- open cemp for select * from emp where deptno=vdeptno; --关闭是close
     open cemp for select ename,job from emp where deptno=vdeptno;
     --遍历游标 loop-end loop
     loop
       --fetch cemp into vrow; 
       fetch cemp into a,b;
       exit when cemp%notfound;--当游标没有数据时退出
       --dbms_output.put_line(vrow.empno||'--'||vrow.ename);  
       dbms_output.put_line(a||'--'||b); 
     end loop;  
end;
c 示例代码运行截图
c.1 输入部门编号前

在这里插入图片描述

c.2 输入部门编号后

在这里插入图片描述

2.3 隐式游标

2.3.1 语法

--sql%rowcount 返回的是最近的一次的受影响行数,如果变动的是单条数据,就可以使用等值比较==
sql%rowcount

2.3.2 示例代码

begin
     delete from emp where empno=10;
     --拿到上一次的操作结果
     delete from emp where empno=7369;
     if sql%rowcount>0 then
        dbms_output.put_line('操作成功'); 
        commit;
     else
        dbms_output.put_line('操作失败');   
        rollback; 
     end if;
end;

2.3.3 示例代码运行截图

在这里插入图片描述

2.4 sql%rowcount小例子练习

2.4.1 思路

对于一个完整的转账业务而言
至少存在两个更新sql语句 也就是有两个受影响行数
可以用两个变量接收sql%rowcount
如果两个变量都大于0,就代表整个业务是完整的,可以通过
否则,需要进行回滚业务

2.4.2 示例代码

declare
  vaccount1 number:=&请输入转出账号;
  vaccount2 number:=&请输入收款账号;
  vmoney number:=100;--转账固定100
  vresult1 number;
  vresult2 number;
begin
  --实现账号1给账号2转账
  update emp set sal=sal-vmoney WHERE empno=vaccount1;
  vresult1:=sql%rowcount;
  update emp set sal=sal+vmoney WHERE empno=vaccount2;
  vresult2:=sql%rowcount;
  --判断是否转账成功
  if vresult1>0 AND vresult2>0 then
     dbms_output.put_line('操作成功');
     commit;
  else
     dbms_output.put_line('操作失败');
     rollback;
  end if;
end;

2.4.3 示例代码运行截图

a 相互转账的账户

在这里插入图片描述

b 控制台截图

在这里插入图片描述

c 更新前

在这里插入图片描述

d 更新后

在这里插入图片描述

3 触发器

3.0 理解

A本质是一个存储过程,发生特定事件时,oracle会执行里面的代码
这里所写的两个例子都是dml(增删改)的触发器
B 写触发器需要注意的是:是在该执行语句前触发,还是在该执行语句之后触发

3.1 语句级触发器

3.1.1 语法

create or replace trigger 触发器名字
before insert or delete or update on emp
--没有for each row,就代表是语句级触发器
begin
   -- 触发语句时执行的代码
end;

3.1.2 示例代码

create or replace trigger tg_empdml
before insert or delete or update on emp
begin
   if to_char(sysdate,'day') in('星期六','星期五') then
      -- raise_application_error 弹出错误弹窗
      raise_application_error(-20888,'周末不上班 搞啥呢');
   end if;
end;

3.1.3 示例代码运行截图
在这里插入图片描述

3.2 行级触发器

3.2.0 语法

CREATE OR replace trigger 触发器名字
-- OR replace指的是存在一样名字的触发器时,是直接去覆盖掉,
--若不写这个,然后又存在相同的,运行就会报错
after/before (新增|删除|修改)
for each row --行级触发器,每一次都触发
declare
  -- 变量定义位置
begin
  -- :new是修改后的值,行级修改值 :old修改之前的值
  --每次触发时执行的语句
end;

3.2.1 三个例子

a 记录表删除、修改的日志信息
a.1 示例代码
CREATE OR replace trigger tg_emplog
after update or delete on emp
for each row --行级触发器,每一次都触发
declare
 vtime varchar2(20):=to_char(sysdate,'yyyy-mm-dd hh24:mi:ss');  
begin
  -- :new是修改后的值,行级修改值 :old修改之前的值
  if updating then
   dbms_output.put_line(vtime||'修改了员工:'||:new.ename);
  --elsif是else..if的意思 
  elsif deleting then
    dbms_output.put_line(vtime||'删除了员工:'||:old.ename); 
  end if;   
end;
a.2 示例代码运行截图

在这里插入图片描述

b 设置员工只能涨薪
b.1 示例代码
create or replace trigger tg_empsal
before update on emp
for each row
declare
begin
  if :new.sal<:old.sal then
     raise_application_error(-20999,'扣我工资,砍死你'); 
  end if;
end;
b.2 示例代码运行截图

在这里插入图片描述

c 实现主键自动递增
c.1 示例代码
CREATE or replace trigger tg_dept
before insert on dept
  for each row when(new.deptno is null) -- 添加 这里不用写冒号 添加部门号是空值就触发
declare 
  vid number;
begin
  select sq_dept.nextval into vid from dual;
  --新设置的编号设置成我的序列值
  :new.deptno:=vid;
end;
c.2 示例代码运行截图

执行前
在这里插入图片描述

执行后
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/96365.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MR案例:学生排序(单字段排序、多字段排序)

文章目录一、提出任务二、完成任务&#xff08;一&#xff09;准备数据1、在虚拟机上创建文本文件2、上传文件到HDFS指定目录&#xff08;二&#xff09;实现步骤1、创建Maven项目2、添加相关依赖3、创建日志属性文件4、创建学生实体类5、创建学生映射器类5、创建学生归并器类6…

JS中操作<select>标签选的值

JS中操作<select>标签选的值 <select>标签是一种表单控件&#xff0c;用来创建下拉列表。在<select> 标签内可用 <option> 标签定义下拉列表中的可用选项。下面给出一个基本下拉列表示例&#xff1a; <!DOCTYPE html> <html lang"zh&q…

Codeforces Round #838 (Div. 2)

A. Divide and Conquer 题目链接&#xff1a;Problem - A - Codeforces 样例输入&#xff1a; 4 4 1 1 1 1 2 7 4 3 1 2 4 1 15样例输出&#xff1a; 0 2 1 4题意&#xff1a;一个数组是好的当且仅当所有的元素和是一个偶数&#xff0c;现在给我们一个初始数组&#xff0c;我…

Android---组件化

1、单体应用 所有代码写在一个工程里。不同业务写到各自模块&#xff0c;以包名来区分。 弊端 1、无论包名做的再好&#xff0c;随着项目扩大&#xff0c;项目失去层次感&#xff0c;接受吃力。 2、报名作为约束&#xff0c;太弱了。一不注意就会出现不同业务之间之间相互调…

【算法数据结构专题】「限流算法专项」带你认识常用的限流算法的技术指南(分析篇)

限流 限流的目的是通过对并发访问/请求进行限速&#xff0c;或者对一个时间窗口内的请求进行限速来保护系统&#xff0c;一旦达到限制速率则可以拒绝服务、排队或等待、降级等处理 限流一词常用于计算机网络之中&#xff0c;定义如下&#xff1a; In computer networks, rate l…

接口测试(七)—— 参数化、数据库操作类封、接口自动化框架

目录 一、接口自动化测试框架 1、目录结构 二、封装iHRM登录 1、普通方式实现 2、登录接口对象层 3、登录接口测试用例层 4、封装断言方法 三、参数化 1、回顾UnitTest参数化 1.1 原始案例 1.2 参数化实现 1.3 从json文件读取 2、登录接口参数化 2.1 组织数据文…

java8流操作之不常用但是很好用的隐藏api

前言 1、一些普通的方式就不再多说了&#xff0c;这里主要说一些不常用的&#xff0c;但是作用很大的api方式 2、如果想要细致了解可以参考 JAVA8的流操作&#xff0c;十分推荐 一、flatMap 1、这个api主要是用来推平流的&#xff0c;和map不一致&#xff0c;map是对象到对…

Python基础(十六):函数的初步认识

文章目录 函数的初步认识 一、函数的作用 二、函数的使用步骤 1、定义函数 2、调用函数 3、快速体验 三、函数的参数作用 四、函数的返回值作用 1、应用 五、函数的说明文档 1、语法 2、快速体验 3、函数嵌套调用 七、函数应用 1、打印图形 2、函数计算 八、总…

还在为多张Excel汇总统计发愁?Python 秒处理真香!

为什么越来越多的非程序员白领都开始学习 Python &#xff1f;他们可能并不是想要学习 Python 去爬取一些网站从而获得酷酷的成就感&#xff0c;而是工作中遇到好多数据分析处理的问题&#xff0c;用 Python 就可以简单高效地解决。本文就通过一个实际的例子来给大家展示一下 P…

新手传奇gm架设要学会的几个修改技巧

每个传奇gm对于架设一个服务器都有自己独立的看法和想法&#xff0c;一些人之所以会想着要架设一个传奇私服主要原因是自己在其他人的服力玩得不是那么舒心。所以想要按照自己的想法和思路打造一个适合自己的专属服务器进行游戏&#xff0c;其实这两者之间是有必然联系的&#…

毕业三年活得像个废物,转行网络安全,写给像我一样迷茫的人...

首先说说我吧&#xff0c;普通二本非科班商贸专业毕业&#xff0c;三年了&#xff0c;做过电商&#xff0c;做过新媒体&#xff0c;做过业务员&#xff0c;从躺平到摆烂&#xff0c;一开始还挺享受这样的生活的&#xff0c;毕竟每月4千工资&#xff0c;抛出吃住&#xff0c;剩个…

重学webpack系列(八) -- webpack的运行机制与工作原理

前面几个章节我们分别去探索了webpack的loader机制、plugin机制、devServer、sourceMap、HMR&#xff0c;但是你是否知道这些配置项是怎么去跟webpack本身执行机制挂钩的呢&#xff1f;这一章我们就来探索一下webpack的运行机制与工作原理吧。 webpack核心工作过程 我们打开w…

第十四章 文件操作

1.文件的基本操作 文件&#xff0c;对我们并不陌生&#xff0c;文件是数据源&#xff08;保存数据的地方&#xff09;的一种&#xff0c;比如大家经常使用的Word文档&#xff0c;TXT文件&#xff0c;excel文件…都是文件。文件最主要的作用就是保存数据&#xff0c;它既可以保…

用户虚拟地址空间管理-mm_struct

一、进程虚拟地址空间管理概览 二、mm_struct结构体的主要成员 atomic_t mm_users;共享同一个用户虚拟地址空间的进程的数量&#xff0c;也就是线程组包含的进程的数量atomic_t mm_count;内存描述符的引用计数struct vm_area_struct *mmap;虚拟内存区域链表struct rb_root mm_…

【java】课程设计--抽卡模拟器

文章目录工期安排自己实现菜单逻辑抽卡算法0.书写要求1.用户需求2.设计思想3.各个功能和算法描述4.系统调试中问题5.总结新知识怎么打开任务管理器改进&#xff08;进一步的设想&#xff09;交给她们实现1.注册登录2.读文件-显示查找内容暂时成功案例工期安排 1 自定义增加和删…

数据权限就该这么设计

在项目实际开发中我们不光要控制一个用户能访问哪些资源&#xff0c;还需要控制用户只能访问资源中的某部分数据。 控制一个用户能访问哪些资源我们有很成熟的权限管理模型即RBAC&#xff0c;但是控制用户只能访问某部分资源&#xff08;即我们常说的数据权限&#xff09;使用…

[附源码]Python计算机毕业设计红旗家具城管理系统Django(程序+LW)

该项目含有源码、文档、程序、数据库、配套开发软件、软件安装教程 项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等…

✿✿✿JavaScript --- 事件

目录 1.事件相关概念 2. js中注册监听&#xff08;事件绑定&#xff09;的方式 (1)在定义标签时&#xff0c;添加事件名称属性。属性值是js代码&#xff08;js代码会被自动封装到一个function函数的方法体中&#xff09; (2)通过js获取元素对象&#xff0c;再添加事件。 补…

ADI Blackfin DSP处理器-BF533的开发详解54:CVBS输出-DSP和CH7024的应用详解(含源码)

硬件准备 ADSP-EDU-BF533&#xff1a;BF533开发板 AD-HP530ICE&#xff1a;ADI DSP仿真器 软件准备 Visual DSP软件 硬件链接 CVBS OUT 视频输出 硬件实现原理 CVBS_OUT 子卡板连接在 ADSP-EDU-BF53x 开发板的扩展端口 PORT3 和 PORT4 上&#xff0c;板卡插入时&#xff0…

window10 下Tomcat安装步骤

目录先安装JDK根据JDK选择tomcat版本下载安装设置系统变量运行测试先安装JDK 略过 根据JDK选择tomcat版本 打开CMD窗口&#xff0c;运行java -version查看本机JDK版本 C:\Users\admin>java -version java version "11.0.17" 2022-10-18 LTS Java(TM) SE Runtime E…