Oracle-动态sql学习笔记,由易至难讲解七个例子

news2025/1/4 19:18:51

本文章的内容来源于对oracle课堂上讲的内容做出的笔记

静态sql和动态sql

·静态sql:

静态 SQL 是在编译时写死的 SQL 语句,即在程序编写阶段,SQL 语句已经被固定下来。

特点:

1.预编译: SQL 语句在程序编译时就会被解析、优化,并生成执行计划,然后存储在数据库中。
2.性能: 由于编译时已经确定了执行计划,因此执行时性能相对较高,因为不需要解析和生成执行计划。
3.安全性: 相对较高,因为 SQL 语句是在编译时确定的,不容易受到 SQL 注入等攻击。

例子:

SELECT * FROM employees WHERE department_id = 10;

·动态sql:

动态 SQL 是在运行时构建和执行的 SQL 语句,允许根据程序运行时的条件来生成不同的 SQL 语句

特点:

1.灵活性: 允许在运行时动态构建 SQL 语句,根据需要进行条件判断、循环等,使得程序更加灵活。
2.执行计划: SQL 语句在运行时解析和生成执行计划,可能会导致一些性能损失,因为这个过程在运行时发生。
3.安全性: 相对较低,因为动态 SQL 可能会受到 SQL 注入攻击,必须谨慎处理输入的参数。

DECLARE
  v_department_id NUMBER := 10;
  v_sql_statement VARCHAR2(100);
BEGIN
  v_sql_statement := 'SELECT * FROM employees WHERE department_id = ' || v_department_id;
  EXECUTE IMMEDIATE v_sql_statement;
END;

什么时候使用动态sql

1.由于在PL/SQL中只能执行静态的查询和DML语句,因此要执行DDL语句或DCL语句,必须借助于动态SQL语句
2.在开发报表或一些复杂的应用程序逻辑时,如果要基于参数化的查询方式,比如动态的表字段和动态的表名称时,可以使用动态SQL语句
3.基于数据表存储业务规则和软件代码,可以将很多的业务规则的代码写在一个表的记录中,在程序需要时检索不同的业务逻辑代码动态的执行

下面用几个例子,从简单到复杂讲解动态sql的几种使用

例子一讲解(基础)

DECLARE
sql_stmt1 VARCHAR2(128);
BEGIN
sql_stmt1:='INSERT INTO rooms 
VALUES (99000, ''Building 7'', 110, 50, ''Discussion Room E'')';
EXECUTE IMMEDIATE sql_stmt1;
COMMIT;
END;

sql_stmt1 被赋予一个包含 INSERT 语句的字符串。这个 INSERT 语句的目的是向名为 rooms 的表中插入一行记录,包括房间号、建筑名称、楼层、容量和房间类型等信息。EXECUTE IMMEDIATE 语句用于在运行时执行动态生成的 SQL 语句。在这里,它执行了 sql_stmt1 中存储的 INSERT 语句,COMMIT是提交事务的命令。在数据库操作中,当你执行一些修改数据的操作时,比如 INSERT、UPDATE 或 DELETE,最后需要使用 COMMIT 语句来确保这些修改永久保存到数据库中

例子二讲解(使用绑定变量)

DECLARE
  v_department_id NUMBER;
  v_department_name VARCHAR2(100);
  v_sql_statement VARCHAR2(200);
BEGIN
  -- 使用 & 符号定义输入参数
  v_department_id := &input_department_id;
  -- 构建动态SQL语句,使用 :department_id 作为绑定变量
  v_sql_statement := 'SELECT department_name FROM departments WHERE department_id = :department_id';
  -- 执行动态SQL语句,并将输入的部门ID传递给绑定变量 :department_id
  EXECUTE IMMEDIATE v_sql_statement INTO v_department_name USING v_department_id;
  -- 输出查询结果
  DBMS_OUTPUT.put_line('部门名称为: ' || v_department_name);
END;

在这个示例中,&input_department_id 是一个替换变量,当代码运行时,系统将提示用户输入一个部门ID。这个输入的部门ID将被赋值给v_department_id,然后用作动态SQL语句中的绑定变量:department_id。这样,在执行动态SQL语句时,用户输入的值将被正确地用于查询操作,最后结果会显示查询到的部门名称。

这种方式对于需要用户交互式输入参数来动态构建SQL语句的情况非常有用,可以根据用户输入灵活地执行查询或其他数据库操作,需要注意的是:

这个地方的冒号以及后面的一堆字符串(遇见空格结束)合起来代表一个占位符,后面这个字符串是可以随意取名的,最后使用USING这个关键字放替代这个占位符的真正的变量,当然,当有多个绑定变量的时候,可以放多个占位符,然后,using后面也可以跟多个变量,用逗号分割

例子三讲解(执行动态PL/SQL块)

DECLARE
plsql_block VARCHAR2(500);
BEGIN
plsql_block := '
BEGIN 
EXECUTE IMMEDIATE ''TRUNCATE TABLE temp_table'';
FOR v_Counter IN 1..50 LOOP
INSERT INTO temp_table
VALUES (v_Counter, ''Loop Index'');
END LOOP; 
END;';
EXECUTE IMMEDIATE plsql_block;
COMMIT;
END;

这个代码执行的内容实际上是plsql_block字符串当中的语句,首先会对temp_table进行TRUNCATE(截断),然后在表中插入50行数据

例子四讲解(使用RETURNING INTO

DECLARE
sql_stmt1 VARCHAR2(500);
v_room_id NUMBER(5):=99994;
v_number_seats NUMBER(4);
BEGIN
sql_stmt1:='UPDATE rooms SET number_seats=number_seats+10'
|| ' WHERE room_id=:room_id RETURNING number_seats 
INTO :number_seats';
EXECUTE IMMEDIATE sql_stmt1 USING v_room_id 
RETURNING INTO v_number_seats;
DBMS_OUTPUT.put_line('调整后的座位数量为:'|| v_number_seats);
COMMIT;
END;

使用 EXECUTE IMMEDIATE 执行存储在 sql_stmt1 中的动态 SQL 语句。通过 USING 子句将 v_room_id 绑定到 :room_id 参数,通过 RETURNING INTO 子句将更新后的 number_seats 返回到 v_number_seats 变量中

例子五讲解(OPEN...FOR处理多行查询)

DECLARE
sql_stmt1 VARCHAR2(500);
TYPE t_RSCurType IS REF CURSOR;
v_rs_cursor t_RSCurType;
v_student_id NUMBER(5):=10000;
v_rs_record registered_students%ROWTYPE;
BEGIN
sql_stmt1 :=' SELECT student_id,department,course,grade FROM 
registered_students WHERE student_id= :student_id';
OPEN v_rs_cursor FOR sql_stmt1 USING v_student_id;
LOOP
FETCH v_rs_cursor INTO v_rs_record;
EXIT WHEN v_rs_cursor%NOTFOUND;
DBMS_OUTPUT.put_LINE('学号:' || v_rs_record.student_id || 
' ' || '系:' ||v_rs_record.department ||' 课程:' || v_rs_record.course 
||' 分数:' || v_rs_record.grade);
END LOOP;
CLOSE v_rs_cursor;
END;

t_RSCurType 是一个游标类型的声明,用于定义 v_rs_cursor 的数据类型。
v_rs_cursor 是一个 REF CURSOR 类型的游标变量。
v_student_id 是一个 NUMBER 类型的变量,赋值为 10000。
v_rs_record 是一个记录类型,用于存储 registered_students 表的一行数据。

使用 OPEN 语句打开游标,并使用 USING 子句将参数 v_student_id 绑定到 SQL 语句中的 :student_id,使用 LOOP 开始循环,通过 FETCH 语句从游标中获取数据,并将其存储到 v_rs_record 变量中。循环会在游标到达结果集的末尾时退出, 使用 CLOSE 语句关闭游标。

EXIT WHEN v_rs_cursor%NOTFOUND意思为搜到结尾就退出

总之,整个代码的作用是查询指定学生的信息并逐行输出。这里使用了动态 SQL,通过参数 v_student_id 绑定到 SQL 语句中,并且游标用于逐行处理查询结果。输出使用 DBMS_OUTPUT 函数,这通常用于在调试时输出信息

例子六讲解(批量绑定)

动态SQL批量绑定指的是在执行动态SQL语句时,一次性处理多个绑定变量值,而不是逐个处理

DECLARE
sql_stmt1 VARCHAR2(500);
TYPE t_student_id IS TABLE OF students.id%TYPE INDEX BY 
BINARY_INTEGER;
TYPE t_first_name IS TABLE OF students.first_name%TYPE INDEX 
BY BINARY_INTEGER;
TYPE t_last_name IS TABLE OF students.last_name%TYPE INDEX 
BY BINARY_INTEGER;
v_student_id_tab t_student_id;
v_first_name_tab t_first_name;
v_last_name_tab t_last_name;
v_major students.major%TYPE:=‘History‘;
BEGIN
sql_stmt1 :=' SELECT student_id,first_name,last_name FROM students 
WHERE major= :major'; 
EXECUTE IMMEDIATE sql_stmt1
BULK COLLECT INTO 
v_student_id_tab,v_first_name_tab,v_last_name_tab
USING v_major;
FOR i IN 1..v_student_id_tab.COUNT LOOP
DBMS_OUTPUT.put_line('学号:' || v_student_id_tab(i) ||' ' || '名:' 
||v_first_name_tab(i) ||' 姓:' || v_last_name_tab(i));
END LOOP;
END;

t_student_id、t_first_name、t_last_name 是三个关联数组的类型定义,索引类型为 BINARY_INTEGER。
v_student_id_tab、v_first_name_tab、v_last_name_tab 是实际的关联数组变量,所谓关联数组,以v_student_id_tab为例,简单来说就是一个数据类型和“students表当中的id”一样的元素的集合

使用 EXECUTE IMMEDIATE 执行动态 SQL 查询,并通过 BULK COLLECT INTO 子句将查询结果批量收集到关联数组中。USING 子句将 v_major 绑定到 SQL 语句中,然后使用 FOR 循环遍历关联数组,逐行输出学生的学号、名字和姓氏

整个代码的作用是查询指定专业的学生信息,并通过关联数组将学号、名字和姓氏分别存储,并通过循环逐行输出。请注意,动态 SQL 的使用使得 SQL 语句在运行时构建,而不是在编译时

例子七讲解(使用RETURNING BULK COLLECT INTO)

DECLARE
sql_stmt1 VARCHAR2(500);
TYPE t_room_id IS TABLE OF rooms.room_id%TYPE INDEX BY 
BINARY_INTEGER;
TYPE t_number_seats IS TABLE OF rooms.number_seats%TYPE 
INDEX BY BINARY_INTEGER;
v_room_id_tab t_room_id;
v_number_seats_tab t_number_seats;
v_building rooms.building%TYPE := 'Music Building';
BEGIN
sql_stmt1:='UPDATE rooms SET number_seats=number_seats+10'
|| ' WHERE building=:building RETURNING 
room_id,number_seats INTO :room_id,:number_seats';
EXECUTE IMMEDIATE sql_stmt1 USING v_building
RETURNING BULK COLLECT INTO 
v_room_id_tab,v_number_seats_tab;
FOR i IN 1..v_room_id_tab.COUNT LOOP
DBMS_OUTPUT.put_line('教室编号:'||v_room_id_tab(i) ||'调整后的
座位数量为:'|| v_number_seats_tab(i));
END LOOP;
COMMIT;
END;

整个代码块的作用是动态地更新符合条件的教室的座位数量,并通过关联数组存储并输出更新后的教室编号和座位数量。请注意,动态 SQL 的使用使得 SQL 语句在运行时构建,而不是在编译时,很多比如说关联数组的用法跟例子六相似,仅仅需要提的是这里使用动态 SQL 构建更新语句,将满足条件的教室的座位数量增加10,并通过 RETURNING INTO 子句将更新后的教室编号和座位数量返回到关联数组中

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

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

相关文章

【刷题专栏—突破思维】142. 环形链表 II

前言:本篇博客将讲解三个OJ题,前两个作为铺垫,最后完成环形链表的节点的寻找 文章目录 一、160. 相交链表二、141. 环形链表三、142. 环形链表II 一、160. 相交链表 题目链接:LeetCode—相交链表 题目描述: 给你两个单…

【VSCode】Visual Studio Code 配置简体中文环境教程

介绍 Visual Studio Code(简称 VS Code)是一款轻量级的代码编辑器,它支持多种编程语言,并且具有丰富的功能和插件扩展。如果你更喜欢使用简体中文界面,那么本教程将向你展示如何在 VS Code 中配置简体中文环境。 步骤…

如何调整图片尺寸:简单实用的教程分享

报名事业编考试的时候,会发现上传照片时会提示图片大小尺寸应该为多少,如果不符合规定就无法提交报名,那么怎么才能修改图片大小呢?最简单的方法就是利用调整照片大小工具来对图片尺寸修改,本文分享一个在线图片处理工…

如何提高领导力?

提升领导力的关键一方面要有正确的自我认知,另一方面就是坚持去实践,没有速成可言,没人那个人天生就具备领导力,有些人我们看起来是具有领导力的,但这些表象的背后可能是家庭环境的潜移默化,也有可能是个人…

智能导诊系统:基于机器学习和自然语言处理技术,可快速推荐合适的科室和医生

智能导诊系统是一种基于人工智能技术的新型系统,它能够为医院提供患者服务和管理,提高医院的管理效率和服务水平。 技术架构:springbootredismybatis plusmysqlRocketMQ 以下是智能导诊系统的应用场景和功能特点: 应用场景 1.患…

一个项目进行测试的一些最基本环境

在一个项目开发到发布的整个过程中,会使用到很多个环境进行测试和运行项目。最基本的开发环境、测试环境、准生产环境、生成环境 一、开发环境 开发环境顾名思义就是我们程序猿自己把项目放到自己的电脑上,配置好以后,跑起来项目&#xff0…

深度学习100例-卷积神经网络(CNN)实现mnist手写数字识别 | 第1天

文章目录 前期工作1. 设置GPU(如果使用的是CPU可以忽略这步)我的环境: 2. 导入数据3.归一化4.可视化5.调整图片格式 二、构建CNN网络模型三、编译模型四、训练模型五、预测六、知识点详解1. MNIST手写数字数据集介绍2. 神经网络程序说明3. 网…

计算机毕业设计 基于Vue的米家商城系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍:✌从事软件开发10年之余,专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ 🍅文末获取源码联系🍅 👇🏻 精…

解决Qt5.13.0无MySQL驱动问题

一、前言 由于Qt5.12.3是最后提供mysql数据库插件的版本,往后的版本需要自行编译对应的mysql数据库插件,官方安装包不再提供。使用高版本的Qt就需要自行编译mysql驱动。 若没有编译在QT中调用Qsqldatabase库连接mysql时,提示出现如下问题&a…

全国矿产地空间分布数据

我国矿产资源总量丰富、品种齐全,但人均占有量不足世界平均水平,矿产资源质量较差,地理分布不均衡,大型、超大型矿和露采矿较少,开发利用率不足,选矿冶炼技术落后。我国铁矿、锰矿、铬铁矿、铜矿、铝土矿等…

YOLOv5独家原创改进:最新原创WIoU_NMS改进点,改进有效可以直接当做自己的原创改进点来写,提升网络模型性能精度

💡该教程为属于《芒果书》📚系列,包含大量的原创首发改进方式, 所有文章都是全网首发原创改进内容🚀 💡本篇文章为YOLOv5独家原创改进:独家首发最新原创WIoU_NMS改进点,改进有效可以直接当做自己的原创改进点来写,提升网络模型性能精度。 💡对自己数据集改进有效…

Postman内置动态参数以及自定义的动态参数

近期在复习Postman的基础知识,在小破站上跟着百里老师系统复习了一遍,也做了一些笔记,希望可以给大家一点点启发。 一)内置动态参数 {{$timestamp}} 生成当前时间的时间戳{{$randomInt}} 生成0-1000之间的随机数{{$guid}} 生成随…

基于平衡优化器算法优化概率神经网络PNN的分类预测 - 附代码

基于平衡优化器算法优化概率神经网络PNN的分类预测 - 附代码 文章目录 基于平衡优化器算法优化概率神经网络PNN的分类预测 - 附代码1.PNN网络概述2.变压器故障诊街系统相关背景2.1 模型建立 3.基于平衡优化器优化的PNN网络5.测试结果6.参考文献7.Matlab代码 摘要:针…

LeetCode - 141. 环形链表 (C语言,快慢指针,配图)

141. 环形链表 - 力扣(LeetCode) 1. 什么是快慢指针 这里我们我们将介绍环形链表的经典解法——快慢指针,简单理解,指针移动快的叫做快指针fast,移动速度慢的叫慢指针slow。一般我们设快指针走两步,慢指针走…

基于单片机的汽车安全气囊系统故障仿真设计

**单片机设计介绍, 基于单片机微波炉加热箱系统设计 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机的汽车安全气囊系统的故障检测系统是一种用于检测安全气囊系统故障的智能化设备,通过单片机控…

js中的instance,isPrototype和getPrototypeOf的使用,来判断类的关系

😁 作者简介:一名大四的学生,致力学习前端开发技术 ⭐️个人主页:夜宵饽饽的主页 ❔ 系列专栏:JavaScript小贴士 👐学习格言:成功不是终点,失败也并非末日,最重要的是继续…

conda安装与镜像源配置

下载Anaconda 最好去下面这个国内镜像网站: https://mirrors.tuna.tsinghua.edu.cn/anaconda/archive/ 下载过程中,我勾选了添加到环境变量(虽然软件爆红提醒了),取消勾选了设置python3.7为默认python版本。 接下来&…

LLM系列 | 27 : 天工大模型Skywork解读及揭露刷榜内幕引发的思考

引言 简介 预训练 ​语料 分词器 模型架构 Infrastructure 训练细节 评测 实战 总结 思考 0. 引言 晨起开门雪满山,雪晴云淡日光寒。 Created by DALLE 3 小伙伴们好,我是《小窗幽记机器学习》的小编:卖热干面的小女孩。紧接前…

并发编程(多线程)-可见性、有序性、原子性问题

可见性 可见性概念 可见性(Visibility):是指一个线程对共享变量进行修改,另一个先立即得到修改后的最新值 可见性演示 案例演示:一个线程根据boolean类型的标记flag,while循环,另一个线程改…

基于鸽群算法优化概率神经网络PNN的分类预测 - 附代码

基于鸽群算法优化概率神经网络PNN的分类预测 - 附代码 文章目录 基于鸽群算法优化概率神经网络PNN的分类预测 - 附代码1.PNN网络概述2.变压器故障诊街系统相关背景2.1 模型建立 3.基于鸽群优化的PNN网络5.测试结果6.参考文献7.Matlab代码 摘要:针对PNN神经网络的光滑…