MYSQL存储过程(一)

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: MYSQL存储过程语法结构:变量及赋值、入参出参、流程控制-判断、流程控制-循环、流程控制-退出、继续循环

1. 环境说明

 MYSQL  5.6

2. 使用说明

 存储过程时数据库的一个重要的对象,可以封装SQL语句集,可以用来完成一些较复杂的业务逻辑,并且可以入参出参(类似于java中的方法的书写)。创建时会预先编译后保存,用户后续调用不需要再次编译。

3. 优缺点

 优点:

在生产环境下,可以通过直接修改存储过程的方式修改业务逻辑(或bug),而不用重启服务器。

执行速度快,存储过程经过编译之后会比单独一条一条执行要快。

减少网络传输流量。

方便优化。

 缺点:

过程化编程,复杂业务处理的维护成本高。

调试不便,无法像JAVA一样断点调试

不同数据库之间可移植性差。(不同数据库语法不一致!)

4. 语法

 官方参考网址

 https://dev.mysql.com/doc/refman/5.6/en/sql-statements.html

 https://dev.mysql.com/doc/refman/5.6/en/sql-compound-statements.html

  • 语法结构
CREATE[DEFINER = user]  PROCEDURE sp_name ([proc_parameter[,...]])[characteristic ...] routine_body
-- proc_parameter参数部分,可以如下书写:[IN| OUT | INOUT ] param_name type
-- type类型可以是MySQL支持的所有类型-- routine_body(程序体)部分,可以书写合法的SQL语句 BEGIN ... END-- 声明结束符。因为MySQL默认使用‘;’作为结束符,而在存储过程中,会使用‘;’作为一段语句的结束,导致‘;’使用冲突delimiter $$
CREATE PROCEDURE hello_procedure ()BEGINSELECT'hello procedure';END $$
call hello_procedure();
  • 变量及赋值
  • 局部变量:用户自定义,在begin/end代码块中有效
语法:
声明变量 declare var_name type [default var_value];举例:declare p_name varchar(32);-- set赋值create procedure p_01()begin  declare p_name varchar(32) default 'unkown';set p_name ='ZS';-- set p_name := 'ls';select p_name;end$$
-- into赋值delimiter $$
create procedure sp_var_into()begin  declare emp_name varchar(32) default 'unkown';  declare emp_no int default 0;select e.empno,e.enameinto emp_no,emp_name from emp e where e.empno=7839;select emp_no,emp_name;end$$
  • 用户变量:用户自定义,当前会话(连接)有效。类比java的成员变量
语法: 
@var_name
不需要提前声明,使用即声明
-- 赋值delimiter $$
create procedure p_02()beginset @p_name ='ZS';-- set p_name := 'ls';end$$
call p_02() $$
select @p_name$$  --可以看到结果
  • 会话变量:由系统提供,当前会话(连接)有效
语法:
@@session.var_nameshow session variables;-- 查看会话变量select @@session.unique_checks;-- 查看某会话变量set @@session.unique_checks=0;--修改会话变量
  • 全局变量:由系统提供,整个mysql服务器有效
语法:
@@global.var_name-- 查看全局变量中变量名有char的记录show global variables like'%char%';-- 查看全局变量character_set_client的值select @@global.character_set_client;
  • 入参出参
-- 语法in| out | inout param_name type
-- IN类型演示delimiter $$
create procedure sp_01(in age int)beginset @user_age = age;end$$
call sp_01(10) $$
select @user_age$$
-- OUT类型,只负责输出!-- 需求:输出传入的地址字符串对应的部门编号。delimiter $$
create procedure sp_02(in loc varchar(64),out dept_no int(11))beginselect d.deptnointo dept_no from dept d where d.loc= loc;--此处强调,要么表起别名,要么入参名不与字段名一致end$$
delimiter ;--测试set @dept_no =100;call sp_01('DALLAS',@dept_no);select @dept_no;-- INOUT类型 delimiter $$
create procedure sp_03(inout name varchar)beginset name = concat('hello',name);end$$
delimiter ;set @user_name ='小明';call sp_03(@user_name);select @user_name;
  • 流程控制-判断
  • IF
-- 语法IF search_condition THEN statement_list
[ELSEIF search_condition THEN statement_list] ...
[ELSE statement_list]END IF
-- 前置知识点:timestampdiff(unit,exp1,exp2) 取差值exp2-exp1差值,单位是unitselect timestampdiff(year,e.hiredate,now())from emp e where e.empno='7499';-- 需求:入职年限<=38是新手 >38并且<=40老员工 >40元老delimiter $$
create procedure sp_hire_if()begin  declare result varchar(32);  if timestampdiff(year,'2001-01-01',now())>40    then set result ='元老';  elseif timestampdiff(year,'2001-01-01',now())>38    then set result ='老员工';  else 
set result ='新手';  end if;select result;end$$
delimiter ;
  • CASE:此语法是不仅可以用在存储过程,查询语句也可以用
-- 语法一(类比java的switch):CASE case_value
    WHEN when_value THEN statement_list
[WHEN when_value THEN statement_list] ...
[ELSE statement_list]END CASE
-- 语法二:CASE
    WHEN search_condition THEN statement_list
[WHEN search_condition THEN statement_list] ...
[ELSE statement_list]END CASE
-- 需求:入职年限年龄<=38是新手 >38并 <=40老员工 >40元老delimiter $$
create procedure sp_hire_case()begin  declare result varchar(32);  declare message varchar(64);  case
    when timestampdiff(year,'2001-01-01',now())>40    then 
set result ='元老';set message ='老年人';  when timestampdiff(year,'2001-01-01',now())>38    then 
set result ='老员工';set message ='中年人';  else 
set result ='新手';set message ='青年';  end case;select result;end$$
delimiter ;
  • 流程控制-循环
  • LOOP
-- 语法[begin_label:] LOOP
    statement_list
END LOOP [end_label]需要说明,loop是死循环,需要手动退出循环,我们可以使用`leave`来退出。
可以把leave看成我们java中的break;与之对应的,就有`iterate`(继续循环)——类比java的continue
--需求:循环打印1到10-- leave控制循环的退出delimiter $$
create procedure sp_flow_loop()begin  declare c_index int default 1;  declare result_str  varchar(256) default '1';  cnt:loop
    if c_index >=10    then leave cnt;    end if;set c_index = c_index +1;set result_str = concat(result_str,',',c_index);  end loop cnt;select result_str;end$$
-- iterate + leave控制循环delimiter $$
create procedure sp_flow_loop02()begin  declare c_index int default 1;  declare result_str  varchar(256) default '1';  cnt:loop
set c_index = c_index +1;set result_str = concat(result_str,',',c_index);    if c_index <10 then 
      iterate cnt;    end if;-- 下面这句话能否执行到?什么时候执行到? 当c_index < 10为false时执行    leave cnt;  end loop cnt;select result_str;end$$
  • repeat
[begin_label:] REPEAT
    statement_list
UNTIL search_condition  -- 直到…为止,才退出循环END REPEAT [end_label]-- 需求:循环打印1到10delimiter $$
create procedure sp_flow_repeat()begin  declare c_index int default 1;-- 收集结果字符串  declare result_str varchar(256) default '1';  count_lab:repeat
set c_index = c_index +1;set result_str = concat(result_str,',',c_index);    until c_index >=10  end repeat count_lab;select result_str;end$$
  • while
类比java的while(){}[begin_label:] WHILE search_condition DO
    statement_list
END WHILE [end_label]-- 需求:循环打印1到10delimiter $$
create procedure sp_flow_while()begin  declare c_index int default 1;-- 收集结果字符串  declare result_str varchar(256) default '1';  while c_index <10 do
set c_index = c_index +1;set result_str = concat(result_str,',',c_index);  end while;select result_str;end$$
  • 流程控制-退出、继续循环
  • leave
-- 退出 LEAVE can be used within BEGIN ... END or loop constructs (LOOP, REPEAT, WHILE).LEAVE label
  • iterate
-- 继续循环 ITERATE can appear only within LOOP, REPEAT, and WHILE statementsITERATE label
相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
25天前
|
存储 SQL NoSQL
|
1月前
|
存储 SQL 关系型数据库
MySql数据库---存储过程
MySql数据库---存储过程
38 5
|
1月前
|
存储 关系型数据库 MySQL
MySQL 存储过程返回更新前记录
MySQL 存储过程返回更新前记录
54 3
|
1月前
|
存储 SQL 关系型数据库
MySQL 存储过程错误信息不打印在控制台
MySQL 存储过程错误信息不打印在控制台
66 1
|
3月前
|
存储 关系型数据库 MySQL
Mysql表结构同步存储过程(适用于模版表)
Mysql表结构同步存储过程(适用于模版表)
48 0
|
3月前
|
存储 SQL 关系型数据库
MySQL 创建存储过程注意项
MySQL 创建存储过程注意项
42 0
|
4月前
|
存储 SQL 关系型数据库
(十四)全解MySQL之各方位事无巨细的剖析存储过程与触发器!
前面的MySQL系列章节中,一直在反复讲述MySQL一些偏理论、底层的知识,很少有涉及到实用技巧的分享,而在本章中则会阐述MySQL一个特别实用的功能,即MySQL的存储过程和触发器。
|
4月前
|
存储 SQL 数据库
MySQL设计规约问题之为什么要避免使用存储过程、触发器和函数
MySQL设计规约问题之为什么要避免使用存储过程、触发器和函数
|
5月前
|
存储 SQL 关系型数据库
MySQL数据库进阶第四篇(视图/存储过程/触发器)
MySQL数据库进阶第四篇(视图/存储过程/触发器)
|
5月前
|
存储 SQL 关系型数据库
MySQL存储过程和存储函数的使用
MySQL的存储过程和存储函数在功能和用法上有明显的区别。存储过程是一组为了完成特定功能的SQL语句集,经编译后存储在数据库中,通过指定名称和参数(如果有)来调用执行,可以返回多个值或结果集,但不直接返回值。而存储函数则是一个有返回值的特殊存储过程,它返回一个值或表对象,可以直接嵌入SQL语句中使用,如SELECT语句中。两者都是为了提高SQL代码的重用性和性能,但使用场景和方式有所不同。
4299 4
下一篇
无影云桌面