您现在的位置是:网站首页> 编程资料编程资料

MYSQL存储过程 注释详解_Mysql_

2023-05-27 907人已围观

简介 MYSQL存储过程 注释详解_Mysql_

0.环境说明:

软件版本
mysql8.0
navicat

1.使用说明

存储过程时数据库的一个重要的对象,可以封装SQL语句集,可以用来完成一些较复杂的业务逻辑,并且可以入参出参(类似于java中的方法的书写)。

​ 创建时会预先编译后保存,用户后续的调用都不需要再次编译。

 // 把editUser类比成一个存储过程 public void editUser(User user,String username){ String a = "nihao"; user.setUsername(username); } main(){ User user = new User(); editUser(user,"张三"); user.getUseranme(); //java基础 } 

大家可能会思考,用sql处理业务逻辑还要重新学,我用java来处理逻辑(比如循环判断、循环查询等)不行吗?那么,为什么还要用存储过程处理业务逻辑呢?

优点:

在生产环境下,可以通过直接修改存储过程的方式修改业务逻辑(或bug),而不用重启服务器。
 执行速度快,存储过程经过编译之后会比单独一条一条执行要快。
 减少网络传输流量。
 方便优化。

缺点:

过程化编程,复杂业务处理的维护成本高。
 调试不便
 不同数据库之间可移植性差。-- 不同数据库语法不一致!

2.准备

数据库参阅资料中的sql脚本:

 delimiter $$ --声明结束符 

3.语法

官方参考网址:

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

 #### 3.0 语法结构 ```sql -- 存储过程结构 CREATE [DEFINER = user] PROCEDURE sp_name ([proc_parameter[,...]]) [characteristic ...] routine_body -- 1. proc_parameter参数部分,可以如下书写: [ IN | OUT | INOUT ] param_name type -- type类型可以是MySQL支持的所有类型 -- 2. routine_body(程序体)部分,可以书写合法的SQL语句 BEGIN ... END 

简单演示:

 -- 声明结束符。因为MySQL默认使用‘;'作为结束符,而在存储过程中,会使用‘;'作为一段语句的结束,导致‘;'使用冲突 delimiter $$ CREATE PROCEDURE hello_procedure () BEGIN SELECT 'hello procedure'; END $$ call hello_procedure(); 

3.1 变量及赋值

类比一下java中的局部变量和成员变量的声明和使用

局部变量:

用户自定义,在begin/end块中有效

语法:
声明变量 declare var_name type [default var_value];
举例:declare nickname varchar(32);

 -- set赋值 delimiter $$ create procedure sp_var01() begin declare nickname varchar(32) default 'unkown'; set nickname = 'ZS'; -- set nickname := 'SF'; select nickname; 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.ename into emp_no,emp_name from emp e where e.empno = 7839; select emp_no,emp_name; end$$

用户变量:
用户自定义,当前会话(连接)有效。类比java的成员变量

语法:
@var_name
不需要提前声明,使用即声明

 -- 赋值 delimiter $$ create procedure sp_var02() begin set @nickname = 'ZS'; -- set nickname := 'SF'; end$$ call sp_var02() $$ select @nickname$$ --可以看到结果 

会话变量:
由系统提供,当前会话(连接)有效

语法:

@@session.var_name

 show 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; 

3.2 入参出参

-- 语法
in | out | inout param_name type

举例:

 -- IN类型演示 delimiter $$ create procedure sp_param01(in age int) begin set @user_age = age; end$$ call sp_param01(10) $$ select @user_age$$ -- OUT类型,只负责输出! -- 需求:输出传入的地址字符串对应的部门编号。 delimiter $$ create procedure sp_param02(in loc varchar(64),out dept_no int(11)) begin select d.deptno into dept_no from dept d where d.loc = loc; --此处强调,要么表起别名,要么入参名不与字段名一致 end$$ delimiter ; --测试 set @dept_no = 100; call sp_param02('DALLAS',@dept_no); select @dept_no; -- INOUT类型 delimiter $$ create procedure sp_param03(inout name varchar(49)) begin set name = concat('hello' ,name); end$$ delimiter ; set @user_name = '小明'; call sp_param03(@user_name); select @user_name; 

3.3 流程控制-判断

官网说明
https://dev.mysql.com/doc/refman/5.6/en/flow-control-statements.html

if

 -- 语法 IF search_condition THEN statement_list [ELSEIF search_condition THEN statement_list] ... [ELSE statement_list] END IF 

举例:

 -- 前置知识点:timestampdiff(unit,exp1,exp2) 取差值exp2-exp1差值,单位是unit select timestampdiff(year,e.hiredate,now()) from emp e where e.empno = '7499 '; delimiter $$ -- DROP PROCEDURE IF EXISTS sp_param04; create procedure sp_param05(in ages timestamp) begin declare result varchar(32); if timestampdiff(year,ages,now())>40 then set result = '元老'; elseif timestampdiff(year,ages,now())>38 then set result = '老员工'; ELSE SET result = '新手'; end if; select result; end $$ delimiter; call sp_param05('1970-02-26 10:00:25'); -- 注意:MYSQL时间戳必须从1970年开始。 

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 ; 

3.4 流程控制-循环

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到10 delimiter $$ create procedure sp_flow_repeat() begin declare c_index int default 1; -- 收集结果字符串 declare result_str varchar(256) default '1'; count_lab:repea
                
                

-六神源码网