数据库中的存储过程和函数

1、定义

存储过程和函数是事先经过编译并存储在数据库中的一段 SQL 语句的集合。

存储过程和函数是在数据库中预先定义并编译好的可复用代码块,可以用于完成特定的任务,如计

算、查询和变换等。

2、好处

提高代码的复用性。

减少数据在数据库和应用服务器之间的传输,提高效率。

减少代码层面的业务处理。

3、存储过程的创建、调用、查看、删除

# 创建存储过程
CREATE PROCEDRE procedure_name([proc_parameter[,...]])
begin--SQL语句
end ;# 调用存储过程
call procedure_name();# 查看存储过程
--查询db_name数据库中的所有的存储过程
select name from mysql.proc where db='db_name';--查询存储过程的状态信息
show procedure status;--查询某个存储过程的定义
show create procedure test.pro_test1 \G;# 删除存储过程
DROP PROCEDURE [IF EXISTS] sp_name ;

案例:创建stu_group()存储过程,封装分组查询总成绩,并按照总成绩升序排序的功能。

DELIMITER $ --修改SQL的结束符
CREATE PROCEDURE stu_group();
beginselect gender, SUM(score) getsum from student group by gender order by getsum ASC;
end $
DELIMITER ; --将SQL的结束符改回
CALL stu_group();

--delimiter
该关键字用来声明SQL 语句的分隔符 , 告诉 MySQL 解释器,该段命令是否已经结束了, mysql是否可以执行了。
默认情况下,delimiter 是分号 ; 。在命令行客户端中,如果有一行命令以分号结束,那么回车后, mysql将会执行该命令。

4、存储过程的语法

(1) 变量

declare:定义一个局部变量,该变量的作用范围只能在BEGIN...END中。

set:直接赋值使用SET,可以赋常量或者赋表达式。

DELIMITER $
CREATE procedure pro_test5()
begindeclare name VARCHAR(20);set name='MYSQL';select name;
end$
DELIMITER ;

也可以使用select ... into方法进行赋值操作。

DELIMITER $
CREATE procedure pro_test5()
begindeclare countnum int;select count(*) into countnum from city;select countnum;
end$
DELIMITER ;

(2)if条件判断

示例:根据定义的身高变量,判断当前身高的所属的身材类型。
180及以上------------身材高挑
170-180--------------标准身材
170以下--------------一般身材

delimiter $
create procedure pro_test6()
begindeclare height int default 175;declare description varchar(50);if height >= 180then set description = '身材高挑';elseif height >= 170 and height < 180then set description = '标准身材';elseset description = '一般身材';end if;select description ;
end$
delimiter ;

(3)传递参数

create procedure pro_name([in/out/inout] 参数名 参数类型)
...
in:该参数可以作为输入,也就是需要调用方传入值,默认
out:该参数作为输出,也就是作为返回值
inout:既可作为输入参数,也可作为输出参数

in输入-------示例:根据定义的身高变量,判断当前身高的所属的身材类型。

delimiter $
create procedure pro_test5(in height int)
begindeclare description varchar(50) default '';if height >= 180then set description = '身材高挑';elseif height >= 170 and height < 180then set description = '标准身材';elseset description = '一般身材';end if;select concat('身高 ',height, '对应的身材类型为:',description) ;
end$
delimiter ;

out输出----- 示例:根据传入的身高变量,获取当前身高的所属的身材类型

delimiter $
create procedure pro_test5(in height int, out description varchar(100))
beginif height >= 180then set description = '身材高挑';elseif height >= 170 and height < 180then set description = '标准身材';elseset description = '一般身材';end if;
end$
delimiter ;
调用
call pro_test5(168, @description);
select @description;

tips:
@description : 这种变量要在变量名称前面加上 “@”符号,叫做用户会话变量,代表整个会话过程他都是有作用的,这个类似于全局变量一样。
@@global.sort_buffffer_size : 这种在变量前加上 "@@" 符号 , 叫做 系统变量。

(4)case结构

示例:给定一个月份,然后计算出所在的季度

delimiter $
create procedure pro_test9(month int)
begindeclare result varchar(20);casewhen month >= 1 and month <= 3then set result = '第一季度';when month >= 4 and month <= 6then set result = '第二季度';when month >= 7 and month <= 9then set result = '第三季度';when month >= 10 and month <= 12then set result = '第四季度';end case;select concat('您输入的月份为:', month,'该月份为 :', result) as content;
end$
delimiter ;

(5)while循环

示例:计算从1加到n的值

delimiter $
create procedure pro_test8(n int)
begindeclare total int default 0;declare num int default 1;while(num <= n) doset total = total + num;set num = num + 1;end while;select total;
end$
delimiter ;

(6)repeat结构

有条件的循环控制语句,当满足条件的时候退出循环。

示例:计算 从1加到n的值

delimiter $
create procedure pro_test20(n int)
begindeclare total int default 0;repeatset total = total + n;set n = n - 1;until n = 0end repeatselect total;
end$
delimiter ;

(7)loop语句

LOOP 实现简单的循环,退出循环的条件需要使用其他的语句定义,通常可以使用 LEAVE 语句实现,具体语法如下:

[begin_label:] LOOP
    statement_list
END LOOP [end_label]

如果不在 statement_list 中增加退出循环的语句,那么 LOOP 语句可以用来实现简单的死循环。

(8)leave语句

用来从标注的流程构造中退出,通常和 BEGIN ... END 或者循环一起使用。

示例:使用 LOOP 和 LEAVE 退出循环。

create procedure pro_test11(n int)
begindeclare total int default 0;ins: LOOPif n <= 0 thenleave ins;end if;set total = total + n;set n = n - 1;end LOOP ins;select total;
end$
delimiter ;

(9) 游标/光标

游标是用来存储查询结果集的数据类型,在存储过程和函数中可以使用光标对结果集进行循环的处理。光标的使用,包括光标的声明、open、fetch和close。

# 声明光标
declare cursor_name cursor for select_statement ;
open光标
open cursor_name ;
fetch光标
fetch cursor_name into var_name[,var_name]...
close光标
close cursor_name ;

# 示例
--auto_increment,自增; comment,给字段或添加注释

create table emp(id int(11) not null auto_increment ,name varchar(50) not null comment '姓名',age int(11) comment '年龄',salary int(11) comment '薪水',primary key('id')
)engine=innodb default charset=utf8 ;insert into emp(id, name, age, salary)values(null,'张三', 55, 3800),(null,'李四', 60, 4000),(null,'王五', 38, 2800),(null,'赵六', 42, 1800)

-- 查询emp表中数据, 并逐行获取进行展示

delimiter$
create procedure pro_test01()
begindeclare e_id int(11);declare e_name varchar(50);declare e_age comment int(11);declare e_salary int(11);declare emp_result cursor for select * from emp;open emp_result;fetch emp_result into e_id, e_name, e_age, e_salary;select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪资为: ',e_salary);fetch emp_result into e_id, e_name, e_age, e_salary;select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪资为: ',e_salary);fetch emp_result into e_id, e_name, e_age, e_salary;select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪资为: ',e_salary);fetch emp_result into e_id, e_name, e_age, e_salary;select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪资为: ',e_salary);close emp_result;
end$
delimiter ;

--通过循环结构 , 获取游标中的数据

delimiter$
create procedure pro_test01()
begindeclare e_id int(11);declare e_name varchar(50);declare e_age comment int(11);declare e_salary int(11);declare has_data int default 1;declare emp_result cursor for select * from emp;declare exit handler for not found set has_data = 0;open emp_result;repeatfetch emp_result into id, name, age, salary;select concat('id=',e_id , ', name=',e_name, ', age=', e_age, ', 薪资为: ',e_salary);until has_data = 0end repeat;close emp_result;
end$
delimiter ;

5、存储函数的语法

CREATE FUNCTION function_name([param type ... ])
RETURNS type
BEGIN
    ...
END;

案例:

定义一个存储函数, 请求满足条件的总记录数

create function count_city(countryId int)
returns int
begindeclare cnum int ;select count(*) into cnum from city where  country_id=countryIdreturn cnum;
end$
delimiter ;
# 调用
select count_city(1);
select count_city(2);

6、区别

(1) 标识符不同,函数标识符是function,存储过程是procedure。

(2) 返回值不同,函数返回单个值或者表对象,而过程没有返回值,但是可以通过out参数返回多个值。

(3) 参数不同,函数的参数只能是IN类型,存储过程的参数可以是IN、OUT、INOUT三种类型。

(4)调用方式不同,函数使用select调用,存储过程需要使用call调用。SQL语句不可调用存储过程。

7、常见使用场景

存储过程:较长复杂的业务逻辑、更新和删除相关操作。

函数 :简单的计算型任务、字符串或日期拼接、返回单个值等。

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

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

相关文章

后端主流框架-SpringMvc-day2

Java中的文件下载 2 文件下载 文件下载&#xff1a;就是将服务器&#xff08;表现在浏览器中&#xff09;中的资源下载&#xff08;复制&#xff09;到本地磁盘&#xff1b; 2.1 前台代码 前台使用超链接&#xff0c;超链接转到后台控制器&#xff0c;在控制器通过流的方式…

AI:111-基于深度学习的工业设备状态监测

🚀点击这里跳转到本专栏,可查阅专栏顶置最新的指南宝典~ 🎉🎊🎉 你的技术旅程将在这里启航! 从基础到实践,深入学习。无论你是初学者还是经验丰富的老手,对于本专栏案例和项目实践都有参考学习意义。 ✨✨✨ 每一个案例都附带有在本地跑过的关键代码,详细讲解供…

Android 车联网——基础简介(一)

传统的车载功能单一,无太多娱乐性,而随着智能化时代的发展,车载系统也被赋予了在系统中预装 Android 应用的能力,基于Android平台的车载信息娱乐系统 —— Android AutoMotive 应运而生。 一、AutoMotive简介 Android Automotive OS 车载操作系统,是一个基本 Android 平台…

zsh配置

install sudo apt update sudo apt install zshsh -c "$(wget https://raw.github.com/ohmyzsh/ohmyzsh/master/tools/install.sh -O -)"chsh -s $(which zsh)安装 zsh-syntax-highlighting 插件&#xff1a; cd ~/.oh-my-zsh/custom/pluginsgit clone https://git…

技术与艺术的完美结合,AI艺术字生成平台一览

在创作的世界里&#xff0c;文字不仅是沟通的桥梁&#xff0c;更是灵魂的画笔。有时&#xff0c;我们的海报需要一丝文案的点缀&#xff0c;一些充满艺术感的文字&#xff0c;为作品注入更多的情感与深度。除了手写的文字&#xff0c;AI艺术字为我们提供了新的可能。AI生成未来…

WiFi7: MLO之概述

原文:MLO允许non-AP 和AP之间发现,认证,关联,建立1个或者多个Link。每个link独立的信道访问和帧交换。每个link的能力通过关联协商。 AP必须设置dot11MultiLinkActivated为真,必须附属于MLD。MLD和其附属的AP(s)必须受本规则限制。 Non-AP 当dot11MultiLinkActivated为真…

CF1909_C. Heavy Intervals题解

CF1909_C. Heavy Intervals题解 题目传送门&#xff08;Problem - C - CodeforcesCodeforces. Programming competitions and contests, programming communityhttps://codeforces.com/contest/1909/problem/C&#xff09;。 题目翻译如下&#xff1a;&#xff08;图片来源&a…

操作系统期末复习!

引论及进程管理 1.操作系统的主要功能有( )。 A.进程管理、存储器管理、设备管理、处理机管理 B.虚拟存储管理、处理机管理、进程管理、文件管理 C.处理机管理、存储器管理、设备管理、文件管理 D.进程管理、中断管理、设备管理、文件管理 2.操作系统是对( )进行管…

Unity3D 如何实现多玩家语音聊天详解

前言 Unity3D是一款强大的跨平台游戏引擎&#xff0c;可以用于开发各种类型的游戏。在多人游戏中&#xff0c;语音聊天是一个非常重要的功能&#xff0c;可以增强玩家之间的交互和沟通。本文将详细介绍如何使用Unity3D实现多玩家语音聊天&#xff0c;并给出技术详解以及代码实…

助力工业生产“智造”,基于YOLOv8全系列模型【n/s/m/l/x】开发构建纺织生产场景下布匹瑕疵检测识别系统

纯粹的工业制造没有办法有长久的发展过程&#xff0c;转制造为全流程全场景的生产智造才是未来最具竞争力的生产场景&#xff0c;在前面的开发实践中我们已经涉足工业生产场景下进行了很多实地的项目开发&#xff0c;如&#xff1a;PCB电路板缺陷检测、焊接缺陷检测、螺母螺钉缺…

elasticsearch7.17.9两节点集群改为单节点

需求 将数据从node-23-1节点中迁移到node-83-1节点。但是现在node-83-1并没有加入到集群中&#xff0c;因此首先将node-83-1加入到node-23-1的集群 解决方案 使用ES版本为7.17.9&#xff0c;最开始设置集群为一个节点&#xff0c;node-23-1的配置如下 cluster.name: my-app…

系列十、Spring Cloud Gateway

一、Spring Cloud Gateway 1.1、概述 Spring Cloud全家桶中有个很重要的组件就是网关&#xff0c;在1.x版本中采用的是Zuul网关&#xff0c;但是在2.x版本中&#xff0c;由于Zuul的升级一直跳票&#xff0c;Spring Cloud最后自己研发了一个网关替代Zuul&#xff0c;即&#xf…

ASP.NET Core基础之图片文件(一)-WebApi图片文件上传到文件夹

阅读本文你的收获&#xff1a; 了解WebApi项目保存上传图片的三种方式学习在WebApi项目中如何上传图片到指定文件夹中 在ASP.NET Core基础之图片文件(一)-WebApi访问静态图片文章中&#xff0c;学习了如何获取WebApi中的静态图片&#xff0c;本文继续分享如何上传图片。 那么…

如果你希望在过滤操作之后清空endorsementIds1集合,你可以使用clear()方法。以下是修改后的代码:

如果你希望在过滤操作之后清空endorsementIds1集合&#xff0c;你可以使用clear()方法。以下是修改后的代码&#xff1a; import java.util.List; import java.util.Set; import java.util.HashSet; // ... 其他代码 ... // 将endorsementIds1转换为Set以提高性能 Set…

详谈电商网站建设的四大流程!

在21世纪的互联网时代&#xff0c;电商网站的建设是每个企业发展不可缺少的一次机遇。企业商城网站建设成功也许会获得更大的了利润&#xff1b;如果网站建设不成功&#xff0c;那么也会带来一定的损失。所以建设电商网站不是那么一件简单的事情。那么电商网站制作流程是怎样的…

算法训练营Day33

#Java #贪心 开源学习资料 Feeling and experiences&#xff1a; 单调递增的数字&#xff1a;力扣题目链接 当且仅当每个相邻位数上的数字 x 和 y 满足 x < y 时&#xff0c;我们称这个整数是单调递增的。 给定一个整数 n &#xff0c;返回 小于或等于 n 的最大数字&am…

C++_find 统计一个单词 在一段文中出现的次数

注解: 使用y.find(x, n)从位置n开始在字符串y中查找子串x首次出现的位置。如果找到了子串x&#xff0c;则find()函数会返回该子串在y中的起始索引&#xff08;位置&#xff09;&#xff0c;否则返回-1&#xff08;或npos&#xff09;表示未找到。当find()函数返回非-1值时&…

我开发了一个聚合网盘资源搜索引擎-支持阿里云盘与夸克网盘资源

还在为找不到电子书资源而发愁&#xff1f;还在愁没有高清影视剧观看&#xff1f; 来试试我开发的云盘资源搜索引擎吧&#xff01; 公众号回复关键词: 搜索 ! 就可以获取到网站网址。 这里还有资源分享微信群&#xff0c;不定期分享资源。 关于界面 怎么使用这个引擎&#x…

ubuntu远程桌面连接之novnc

一、前言 该操作是为了实现vnc桌面连接为url连接方式&#xff0c;且在浏览器中可以对ubuntu进行操作。在使用novnc进行操作前&#xff0c;需要先安装vnc才可。ubuntu下如何安装vnc&#xff0c;可看博主前面写的一篇文&#xff0c;ubuntu远程桌面连接之vnc-CSDN博客&#xff0c;…

[雷池WAF]长亭雷池WAF配置基于健康监测的负载均衡,实现故障自动切换上游服务器

为了进一步加强内网安全&#xff0c;在原有硬WAF的基础上&#xff0c;又在内网使用的社区版的雷池WAF&#xff0c;作为应用上层的软WAF。从而实现多WAF防护的架构。 经过进一步了解&#xff0c;发现雷池WAF的上游转发代理是基于Tengine的&#xff0c;所以萌生出了一个想法&…