Oracle存储过程~封神之作

news2025/1/23 21:15:13

简介

Oracle 存储过程是 Oracle 数据库中的一种数据处理对象,它可以在数据库中定义一组预定义的 SQL 语句,用于完成特定的数据库操作。存储过程可以被授权的用户调用,并且可以执行多个语句,这些语句可以被视为一个单独的操作,也可以被视为一系列的操作。

使用存储过程可以大大提高数据库的性能和安全性。存储过程可以减少网络流量和请求,同时也可以减少与应用程序之间的接口调用,从而提高了数据库的性能和可靠性。

目录

基本语法, 游标,存储过程。

1、基本语法 

1.1 新建测试窗口

  • New   Test Window 

 

1.2 程序结构

  • 在Java编程中是区分大小写,这里不用区分大小写。
  • DECLARE部分声明变量或游标(结果集类型变量),程序没有变量声明的可以省略或删除。
  • PLSQL可分为三个部分:  变量声明部分,执行部分,异常处理部分。
-- Created on 2023/5/10 by 肖 
declare 
  --声明变量 游标

begin
  -- 执行语句
  
  -- 异常处理
end;

 1.3打印输出

  •  Dbms_Output 为oracle内置程序包,类似Java中的System.out,而put_line() 是调用的方法,相当于println()方法。
  • 需要注意的是:  put_line('hello mr.xiao'); 中  一定是 ' '  号,附则会报错的。
begin
  
  --打印 hello mr.xiao
 Dbms_Output.put_line('hello mr.xiao');
 
end;

 执行结果

  • 如果你不能打印输出,需要开启  set serveroutput on  因为 默认情况下,输出选项是关闭状态。

1.4 变量

变量分两大类如:

  • 普通数据类型(char,varchar2, date, number, boolean, long)
  • 特殊变量类型(引用型变量、记录型变量)

声明变量的方式如:

  • 变量名  变量类型(变量长度)  例如: v_name  varchar2(30);

1.4.1 普通变量

变量赋值的方式有两种如:

  • 直接赋值语句     :=      比如:    v_name  := '你才是臭弟弟'
  • 语句赋值,使用select …into … 赋值:(语法 select 值 into 变量)
  -- 打印个人信息,包括: 姓名、薪水、地址
DECLARE
  -- 姓名
  V_NAME VARCHAR2(30) := '你才是臭弟弟'; -- 声明变量直接赋值
  --薪水
  V_SAL NUMBER;
  --地址
  V_ADDR VARCHAR2(200);

BEGIN

  --在程序中直接赋值
  V_SAL := 1800; --工资每月1800 每天笑哈哈

  --语句赋值
  SELECT 'CSDN你才是臭弟弟' INTO V_ADDR FROM DUAL; --不会有人不知道DUAL吧,DUAL 是一个用于描述 Oracle 数据库中的虚拟表的关键字

  --打印变量  注意 || 是拼接
  DBMS_OUTPUT.PUT_LINE('姓名:' || V_NAME || ',薪水:' || V_SAL || ',地址:' ||V_ADDR);

END;

执行结果:

1.4.2 引用型变量

  • 变量的类型和长度取决于表中字段的类型和长度
  • 通过 表名.列名%TYPE 指定变量的类型和长度,例如: v_name  emp.ename%TYPE
  -- 查询emp表中1001号员工的个人信息,打印姓名和薪水
DECLARE
  -- 姓名
  V_NAME EMP.ENAME%TYPE; -- 声明变量直接赋值
  --薪水
  V_SAL  EMP.ESALARY%TYPE;

BEGIN
  --查询表中的姓名和薪水并赋值给变量
  --注意查询的字段和赋值的变量的顺序、个数、类型要一致
  SELECT ENAME, ESALARY INTO V_NAME, V_SAL FROM EMP WHERE EMPLOYEEID = 1001;

  --打印变量
  DBMS_OUTPUT.PUT_LINE('姓名:' || V_NAME || ',薪水:' || V_SAL);

END;

执行结果:

推荐大家使用引用型变量区别:

  •   普通型变量:  V_NAME VARCHAR2(30); 你怎么知道一定VARCHAR2类型,你又怎么知道长度一定是30呢,假设V_NAME VARCHAR2(1);  就对应不上SELECT ENAME, ESALARY INTO V_NAME, V_SAL FROM EMP 这条语句中的 ENAME 的长度了,也就接收不到值了,就会报错。如果要使用普通变量前提是 ,了解查询表中对应字段的 类型 及 长度, 才能基于他们来确定类型长度 这比较繁琐。
  •   引用型变量:  声明一个变量 不再定义类型长度,而是基于接收表字段的类型及长度 来定义。如:V_NAME EMP.ENAME%TYPE;

总结:

 使用普通变量定义方式,需要知道表中列的类型,而使用引用类型,不需要考虑列的类型,使用%TYPE是非常好的编程风格,因为引用型变量更加灵活。

1.4.3 记录型变量

  •  记录型变量 接受表中的一整行记录,相当于Java中的一个对象
  • 语法: 变量名称   表名%ROWTYPE, 例如:v_emp  emp%rowtype;
  -- 查询emp表中1001号员工的个人信息,打印姓名和薪水
DECLARE
  -- 记录型变量接受一行
  V_EMP EMP%ROWTYPE;

BEGIN
  --记录型变量默认接受表中的一行数据,不能指定字段。
  SELECT * INTO V_EMP FROM EMP WHERE EMPLOYEEID = 1001;

  --打印变量,通过变量名.属性的方式获取变量中的值
  DBMS_OUTPUT.PUT_LINE('姓名:' || V_EMP.ENAME || ',薪水:' || V_EMP.ESALARY);

END;

总结:

  • 如果有一张表,有50个字段,那么你程序如果要使用这50字段话,如果你使用引用型变量一个个声明,会特别繁琐,记录型变量可以方便的解决这个问题。

注意错误的使用案例如下:

  • 记录型变量只能存储一个完整的行数据

我把 * 换成 单个字段执行报错,因为上面的变量定义的是一行,而现在只给一个是不行的。

  • 返回的行太多了,记录型变量也接收不了

 现在这条sql 为什么报错,因为现在是全表查询 返回的行数超出了 一个变量只能接收一行,这就和JAVA 类似了,应该用集合去装才可以 ,装进集合在取出来  是不是就跟JAVA 中循环取值差不多。

1.5 流程控制

1.5.1 条件分支

  • IF条件判断~语法

BEGIN

  IF 条件 THEN
      执行语句
  END IF;

END;
  • IF...ELSE 条件判断~语法
BEGIN

 IF 条件 THEN
    执行语句
 ELSE
    执行语句
 END IF;

END;
  • IF...ELSIF...ELSE条件判断~语法,注意关键字:ELSIF。
BEGIN

  IF 条件1 THEN 执行1
    
   ELSIF 条件2 THEN 执行2
  
   ELSE 执行3
    
  END IF; 
  
END;

案例:

  --判断emp表中记录是否超过20条,10-20之间,或者10条以下
DECLARE
  --声明变量接受emp表中的记录数
  V_COUNT NUMBER;

BEGIN

  --查询emp表中的记录数赋值给变量

  SELECT COUNT(1) INTO V_COUNT FROM EMP;

  --判断打印

  IF V_COUNT > 20 THEN
    DBMS_OUTPUT.PUT_LINE('EMP表中的记录数超过了20条为:' || V_COUNT || '条。');

  ELSIF V_COUNT >= 10 THEN
    DBMS_OUTPUT.PUT_LINE('EMP表中的记录数在10~20条之间为:' || V_COUNT || '条。');

  ELSE
    DBMS_OUTPUT.PUT_LINE('EMP表中的记录数在10条以下为:' || V_COUNT || '条。');

  END IF;

END;

执行结果:

1.5.2 循环

  • Loop 语法
BEGIN
  LOOP
  	EXIT WHEN 退出循环条件  
  END LOOP;
END;

Loop语法 案例:

  --循环打印 1-5
DECLARE
  --声明循环变量并赋初值
  V_NUM NUMBER := 1;

BEGIN

  LOOP
  
    EXIT WHEN V_NUM > 5;
    
    DBMS_OUTPUT.PUT_LINE(V_NUM);
  
    --循环变量自增
    V_NUM := V_NUM + 1;
  
  END LOOP;

END;

  • While 语法
while(判断循环的条件) loop
	循环的语句;
END loop;

While语法 案例:

DECLARE
    --声明循环变量
    V_NUM NUMBER;
BEGIN
    -- 必须给一个初始值
    V_NUM := 1;
    WHILE(V_NUM < 10) LOOP
        DBMS_OUTPUT.put_line('值为: ' || V_NUM);
        V_NUM := V_NUM + 1;
    END LOOP;
END;

--此循环会先判断再执行语句

  •  FOR循环 语法
FOR 变量名 in 变量的初始值..结束值 lOOP
	循环语句;
END loop;

FOR循环语法 案例:

    --for循环打印 1-10
DECLARE
    --声明循环变量并赋初值
    V_NUM NUMBER ;
BEGIN
	  --此语句会自动将1到10赋值给V_NUM
    FOR V_NUM in 1..10 loop
        DBMS_OUTPUT.put_line('值为: ' || V_NUM);
    END LOOP;
END;

2、游标

2.1 游标说明

  • 用于临时存储一个查询返回的多行数据,通过遍历游标,可以逐行访问处理该结果集的数据。
  • 游标的使用方式:声明→打开→读取→关闭

2.2 语法

游标声明:

CURSOR  游标名[(参数列表)]    IS 查询语句;

游标的打开:

OPEN 游标名;

游标的取值:

FETCH 游标名 INTO 变量列表;

游标的关闭:

CLOSE 游标名;

 注意: 游标名自身是可以带参数的,如果有参数、参数会带入到查询语句中进行查询,游标本质 就是 一个 is 查询语句,也就是说查询结果被放置到游标中。

2.3 游标属性

游标的属性

属性 说明 %FOUND 变量最后从游标中获取记录的时候,在结果集中找到了记录。 %NOTFOUND 变量最后从游标中获取记录的时候,在结果集中没有找到记录。 %ROWCOUNT 当前时刻已经从游标中获取的记录数量。 %ISOPEN 是否打开。 %ROW 游标指向的行数。 %COLUMN 游标指向的列数。 %ATTEMPTS 尝试获取记录的次数。 %ERROR 发生错误的次数。 %FETCH_STATUS FETCH语句的执行状态,包括成功、失败和出错标志。 %SIZE 当前游标指向的记录大小。 %LINE_NUMBER 当前行号。 %ERROR_STRING 错误信息字符串。 %PROCID 当前执行的SQL语句的ID

2.4 无参数游标

  • 使用游标查询emp表中所有员工的姓名和工资,Loop循环依次打印结果集。
--使用游标查询emp表中所有员工的姓名和工资,依次打印结果集。
DECLARE
  --声明游标
  CURSOR C_EMP IS
    SELECT ENAME, ESALARY FROM EMP;

  --声明变量用来接受游标中的元素
  V_ENAME EMP.ENAME%TYPE;

  V_SAL EMP.ESALARY%TYPE;

BEGIN

  --打开游标
  OPEN C_EMP;

  --遍历游标中的值
  LOOP
  
    --通过FETCH语句获取游标中的值并赋值给变量
    FETCH C_EMP
      INTO V_ENAME, V_SAL;
  
    --通过%NOTFOUND判断是否有值,有值打印,没有则退出循环
    EXIT WHEN C_EMP%NOTFOUND;
  
    DBMS_OUTPUT.PUT_LINE('姓名:' || V_ENAME || ',薪水:' || V_SAL);
  
  END LOOP;

  --关闭游标
  CLOSE C_EMP;

END;

执行结果:

2.5 带参数的游标

  • 使用游标查询并打印某部门的员工的姓名和薪资,部门编号为运行时手动输入。
 --使用游标查询并打印某部门的员工的姓名和薪资,部门编号为运行时手动输入。
DECLARE
  --声明游标传递参数
  CURSOR C_EMP(V_EMPLOID EMP.EMPLOYEEID%TYPE) IS
    SELECT ENAME, ESALARY FROM EMP WHERE EMPLOYEEID = V_EMPLOID; 

  --声明变量用来接受游标中的元素
  V_ENAME EMP.ENAME%TYPE;

  V_SAL EMP.ESALARY%TYPE;

BEGIN

  --打开游标并传递参数
  OPEN C_EMP(1001);

  --遍历游标中的值
  LOOP

  --通过FETCH语句获取游标中的值并赋值给变量
    FETCH C_EMP
      INTO V_ENAME, V_SAL;


       --通过%NOTFOUND判断是否有值,有值打印,没有则退出循环
       EXIT WHEN C_EMP%NOTFOUND;
       
       

    DBMS_OUTPUT.PUT_LINE('姓名:' || V_ENAME || ',薪水:' || V_SAL);


  END LOOP;

  --关闭游标
  CLOSE C_EMP;

END;

执行结果:

 注意:%NOTFOUND属性默认值为FLASE,所以在循环中要注意判断条件的位置.如果先判断在FETCH会导致最后一条记录的值被打印两次(多循环一次默认);

错误反例演示:

 反例执行结果:

原因:  %NOTFOUND 默认值是 false,false意味着游标里面默认是有值,到底有值还是没值 需要fetch 好之后才知道有没有值,%NOTFOUND 默认做了一个有值的假设 , 看下面代码:

LOOP


       --通过%NOTFOUND判断是否有值,有值打印,没有则退出循环
       EXIT WHEN C_EMP%NOTFOUND;


     --通过FETCH语句获取游标中的值并赋值给变量
     FETCH C_EMP
        INTO V_ENAME, V_SAL;
       

      DBMS_OUTPUT.PUT_LINE('姓名:' || V_ENAME || ',薪水:' || V_SAL);


END LOOP;

 EXIT WHEN C_EMP%NOTFOUND; 判断有值打印,出去之后 又带着有值的进入到循环中 EXIT WHEN C_EMP%NOTFOUND; 判断没值打印, 所以这次打印的是上次值的。

3、存储过程

3.1 创建存储过程

    持续更新中........................

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

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

相关文章

【逗老师的无线电】快速记录一下MMDVM串口屏相关

最近在研究MMDVM的串口屏&#xff0c;设计知识点比较多&#xff0c;本文先随手记一下&#xff0c;同时随时更新&#xff0c;最后形成完整的文档 一、Nextion和国内串口屏设计 MMDVM默认对接的是Nextion屏幕。但是有人跟我说&#xff0c;NXT的屏幕和国内陶晶驰的屏幕就是一个…

每周一算法:差分算法

差分算法 差分是一种常见的算法&#xff0c;用于快速修改数组中某一段区间的值。其基本思想就是预处理出数组的差分数组&#xff0c;然后修改区间时&#xff0c;只需要修改两个位置的值&#xff0c;即可快速完成区间修改。最后再通过差分数组求出原数组。差分算法在区间加、区…

kafka安装及配置

1. 下载 下载地址&#xff1a;Apache Kafka 我这里下载的是 3.2.1 版本。 2. 上传并解压 上传到 linux 下的 /home/software/ 目录下&#xff0c;然后解压 kafka_2.13-3.2.1.tgz 包到/usr/local/ cd /home/software tar -zxvf kafka_2.13-3.2.1.tgz -C /usr/local # -C 选…

【机器学习】信息量、香农熵、信息增益(增加例子,方便理解)

这节可以搭配 【机器学习】Logistic回归&#xff08;重新整理&#xff09;信息量&#xff08;信息&#xff09;信息量公式的推理过程 香农熵信息增益 【机器学习】Logistic回归&#xff08;重新整理&#xff09; B站视频&#xff1a;“交叉熵”如何做损失函数&#xff1f;打包…

医学影像系统源码,三维后处理和重建 PACS源码

医学影像系统源码&#xff0c;三维后处理和重建 PACS源码 医学影像系统由PACS系统、RIS系统组成&#xff0c;提供与HIS的接口&#xff08;HL7或其他类型&#xff09;。 主要功能介绍 信息预约登记 支持对患者、检查项目、申请医生、申请单据、设备等信息进行管理。且支持检查…

Apache网站部署SSL证书开启https访问,强制http跳转https

centos服务器博客网站安装教程可见&#xff1a;Centos 7.X WordPress博客网站详细教程 FTP/PHP/mysql/Apache环境构建_autotian的博客-CSDN博客本文详细的介绍了centos7.x系统下&#xff0c;如何构建FTP、mysql、PHP、Apache环境&#xff0c;并成功安装WordPress博客网站。http…

Java—JDK8新特性—函数式接口【内含思维导图】

目录 3.函数式接口 思维导图 3.1 什么是函数式接口 3.2 functionalinterface注解 源码分析 3.3 Lambda表达式和函数式接口关系 3.4 使用函数式接口 3.5 内置函数式接口 四大核的函数式接口区别 3.5.1 Supplier 函数式接口源码分析 3.5.2 Supplier 函数式接口使用 3.…

基于AT89C51单片机的篮球计时记分设计

点击链接获取Keil源码与Project Backups仿真图: https://download.csdn.net/download/qq_64505944/87771065 源码获取 主要内容: 基于51单片机设计篮球计时计分器,结合单片机串行接口原理,用AT89C51设计一个篮球比赛计分计时器,能够通过数码管显示分数和比赛时间(并设有…

linux驱动-gpio

最近处理es8336声卡问题&#xff0c;最后排查是spk_ctl_gpio和hp_det_gpio这两个gpio导致的&#xff0c;所以恶补了一下gpio相关的知识&#xff0c;现在总结一下。 源代码使用的是飞腾的gitee上开源的内核&#xff1a;https://gitee.com/phytium_embedded/phytium-linux-kernel…

什么是SVG格式?如何制作?

图像质量对页面非常重要——扭曲和缩放变形的标志、图标或照片会使页面看起来粗糙和不协调&#xff0c;这个问题只会因为响应设计而复杂。 访问者通过桌面机和智能手机查看应用程序&#xff0c;因此无论使用什么设备&#xff0c;图像都应该进行优化。如果有一个数字格式可以让…

基于FPGA+JESD204B 时钟双通道 6.4GSPS 高速数据采集设计(三)连续多段触发存储及传输逻辑设计

本章将完成数据速率为 80MHz 、位宽为 12bits 的 80 路并行采样数据的连续多 段触发存储。首先&#xff0c;给出数据触发存储的整体框架及功能模块划分。然后&#xff0c;简介 MIG 用户接口、设置及读写时序。最后&#xff0c;进行数据跨时钟域模块设计&#xff0c;内存…

XC7VX690T PCIE 硬件设计注意事项

首先参考PG023找到对应封装支持的位置 然后参考UG476找到对应的实际物理位置 XILINX 也有给出对应的推荐位置

gitlab服务器发送邮件配置

1.修改gitlab的配置文件&#xff1a; vim /etc/gitlab/gitlab.rb 这里具体的gitlab.rb文件所在路径需要根据实际的来 找到如下图所示的部分&#xff0c;放开注释&#xff0c;修改配置&#xff0c;此处我用的发件邮箱是QQ邮箱&#xff0c;所以域名配置都是qq.com&#xff0c;…

Springboot application/yaml/yml没有提示解决方案

有下面提示别安装该插件 卸载wl spring assistant plugin就好了,标志是yml图标变成六边形

线程间互斥-mutex互斥锁和lock_guard

要点 锁双重判断的技法 竟态条件&#xff1a;多线程程序执行的结果一致&#xff0c;不会随着CPU对线程不同的调用顺序 线程间安全实例——3个窗口同时卖票 线程不安全的代码如下 int ticketCount 100; // 100张车票 // 模拟10个窗口同时卖票 void sellTicket(int index) …

PostgreSQL11 | 索引

截止到上一篇《PostgreSQL11 | 查询数据》属于pgsql的基础部分就算是都总结完了&#xff0c;从这一篇&#xff08;第9章&#xff09;开始一直到本专栏最后一篇文章&#xff08;第14章&#xff09;都是进阶部分&#xff0c;sql量会减弱&#xff0c;抽象的概念会越来越多&#xf…

数字识别问题

文章目录 6.1 MNIST数据处理6.2.1 训练数据6.2.2 变量管理6.3.1 保存模型6.3.1 加载计算图6.3.1 加载模型6.3.2 导出元图 6.1 MNIST数据处理 在直接在第6章的目录下面创建文件 compat.v1.是tensorflow2.x的语法&#xff0c;全部删掉 删除compat.v1.后的代码 # -*- coding: …

【SCI一区】考虑P2G和碳捕集设备的热电联供综合能源系统优化调度模型(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

学术必备的21个论文网站,建议收藏!

1、综合型论文网站&#xff08;国内&#xff09; &#xff08;1&#xff09;知网 介绍&#xff1a;国内知名度最高的网站&#xff0c;拥有上亿篇各种论文期刊&#xff0c;包含中国学术文献、 外文文献、学位论文、报纸、会议、年鉴、工具书等各类资源统一检索、统一导 航、…

第四章 图像的形态学操作

文章目录 前言一、阈值控制二、腐蚀与膨胀1.腐蚀2.膨胀3.形态学操作 总结 前言 前面讲解了图像基础理论、图像的变换以及图像滤波等操作&#xff0c;本章&#xff0c;将会介绍图像的形态学操作。 图像的形态学指的是一组数学方法和工具&#xff0c;用于图像分析和处理。形态学…