2000字范文,分享全网优秀范文,学习好帮手!
2000字范文 > MySQL存储过程从入门到精通

MySQL存储过程从入门到精通

时间:2021-05-18 06:47:12

相关推荐

MySQL存储过程从入门到精通

文章目录

0.环境说明1.使用说明2.准备3.语法3.0 语法结构3.1 变量及赋值1)局部变量2)用户变量3)会话变量4)全局变量3.2 入参出参3.3 流程控制-判断`if``case`3.4 流程控制-循环`loop``repeat``while`3.5 流程控制-退出、继续循环`leave``iterate`3.6 游标3.7 存储过程中的handler4.练习4.1 利用存储过程更新数据4.2 循环创建表4.3 其他场景5.其他5.1 characteristic5.2 死循环处理5.3 可以在select语句中写case5.4 临时表5.5 复制表和数据

0.环境说明

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.语法

-- 官方参考网址/doc/refman/5.6/en/sql-statements.html/doc/refman/5.6/en/sql-compound-statements.html

3.0 语法结构

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();

3.1 变量及赋值

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

1)局部变量

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

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

-- set赋值create procedure sp_var01()begindeclare nickname varchar(32) default 'unkown';set nickname = 'ZS';-- set nickname := 'SF';select nickname;end$$

-- into赋值delimiter $$create procedure sp_var_into()begindeclare 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$$

2)用户变量

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

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

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

3)会话变量

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

语法:@@session.var_name

show session variables; -- 查看会话变量select @@session.unique_checks; -- 查看某会话变量set @@session.unique_checks = 0; --修改会话变量

4)全局变量

由系统提供,整个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)beginset @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))beginselect d.deptno into dept_no from dept d where d.loc = loc;--此处强调,要么表起别名,要么入参名不与字段名一致end$$delimiter ;--测试set @dept_no = 100;call sp_param01('DALLAS',@dept_no);select @dept_no;

-- INOUT类型 delimiter $$create procedure sp_param03(inout name varchar)beginset name = concat('hello' ,name);end$$delimiter ;set @user_name = '小明';call sp_param03(@user_name);select @user_name;

3.3 流程控制-判断

官网说明:/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差值,单位是unitselect timestampdiff(year,e.hiredate,now()) from emp e where e.empno = '7499';

-- 需求:入职年限<=38是新手 >38并且<=40老员工 >40元老delimiter $$create procedure sp_hire_if()begindeclare result varchar(32);if timestampdiff(year,'2001-01-01',now()) > 40 then set result = '元老';elseif timestampdiff(year,'2001-01-01',now()) > 38then set result = '老员工';else set result = '新手';end if;select result;end$$delimiter ;

case

此语法是不仅可以用在存储过程,查询语句也可以用!

-- 语法一(类比java的switch):CASE case_valueWHEN when_value THEN statement_list[WHEN when_value THEN statement_list] ...[ELSE statement_list]END CASE-- 语法二:CASEWHEN 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()begindeclare result varchar(32);declare message varchar(64);casewhen timestampdiff(year,'2001-01-01',now()) > 40 then set result = '元老';set message = '老爷爷';when timestampdiff(year,'2001-01-01',now()) > 38then set result = '老员工';set message = '油腻中年人';else set result = '新手';set message = '萌新';end case;select result;end$$delimiter ;

3.4 流程控制-循环

loop

-- 语法[begin_label:] LOOPstatement_listEND LOOP [end_label]

举例

需要说明,loop是死循环,需要手动退出循环,我们可以使用leave来退出。

可以把leave看成我们java中的break;与之对应的,就有iterate(继续循环)——类比java的continue

--需求:循环打印1到10-- leave控制循环的退出delimiter $$create procedure sp_flow_loop()begindeclare c_index int default 1;declare result_str varchar(256) default '1';cnt:loopif c_index >= 10then 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()begindeclare c_index int default 1;declare result_str varchar(256) default '1';cnt:loopset 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

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-3xOCz6YT-1633920266336)(assets/1586956047152.png)]

[begin_label:] REPEATstatement_listUNTIL search_condition-- 直到…为止,才退出循环END REPEAT [end_label]

-- 需求:循环打印1到10delimiter $$create procedure sp_flow_repeat()begindeclare c_index int default 1;-- 收集结果字符串declare result_str varchar(256) default '1';count_lab:repeatset c_index = c_index + 1;set result_str = concat(result_str,',',c_index);until c_index >= 10end repeat count_lab;select result_str;end$$

while

类比java的while(){}

[begin_label:] WHILE search_condition DOstatement_listEND WHILE [end_label]

-- 需求:循环打印1到10delimiter $$create procedure sp_flow_while()begindeclare c_index int default 1;-- 收集结果字符串declare result_str varchar(256) default '1';while c_index < 10 doset c_index = c_index + 1;set result_str = concat(result_str,',',c_index);end while;select result_str;end$$

3.5 流程控制-退出、继续循环

leave

类比java的breake

-- 退出 LEAVE can be used within BEGIN ... END or loop constructs (LOOP, REPEAT, WHILE).LEAVE label

iterate

类比java的continue

-- 继续循环 ITERATE can appear only within LOOP, REPEAT, and WHILE statementsITERATE label

3.6 游标

用游标得到某一个结果集,逐行处理数据。

类比jdbc的ResultSet

-- 声明语法DECLARE cursor_name CURSOR FOR select_statement-- 打开语法OPEN cursor_name-- 取值语法FETCH cursor_name INTO var_name [, var_name] ...-- 关闭语法CLOSE cursor_name

-- 需求:按照部门名称查询员工,通过select查看员工的编号、姓名、薪资。(注意,此处仅仅演示游标用法)delimiter $$create procedure sp_create_table02(in dept_name varchar(32))begindeclare e_no int;declare e_name varchar(32);declare e_sal decimal(7,2);declare lp_flag boolean default true;declare emp_cursor cursor for select e.empno,e.ename,e.salfrom emp e,dept dwhere e.deptno = d.deptno and d.dname = dept_name;-- handler 句柄declare continue handler for NOT FOUND set lp_flag = false;open emp_cursor;emp_loop:loopfetch emp_cursor into e_no,e_name,e_sal;if lp_flag thenselect e_no,e_name,e_sal;elseleave emp_loop;end if;end loop emp_loop;set @end_falg = 'exit_flag';close emp_cursor;end$$call sp_create_table02('RESEARCH');

特别注意:

在语法中,变量声明、游标声明、handler声明是必须按照先后顺序书写的,否则创建存储过程出错。

3.7 存储过程中的handler

DECLARE handler_action HANDLERFOR condition_value [, condition_value] ...statementhandler_action: {CONTINUE| EXIT| UNDO}condition_value: {mysql_error_code| SQLSTATE [VALUE] sqlstate_value| condition_name| SQLWARNING| NOT FOUND| SQLEXCEPTION}CONTINUE: Execution of the current program continues.EXIT: Execution terminates for the BEGIN ... END compound statement in which the handler is declared. This is true even if the condition occurs in an inner block.SQLWARNING: Shorthand for the class of SQLSTATE values that begin with '01'.NOT FOUND: Shorthand for the class of SQLSTATE values that begin with '02'.SQLEXCEPTION: Shorthand for the class of SQLSTATE values that do not begin with '00', '01', or '02'.

-- 各种写法:DECLARE exit HANDLER FOR SQLSTATE '42S01' set @res_table = 'EXISTS';DECLARE continue HANDLER FOR 1050 set @res_table = 'EXISTS';DECLARE continue HANDLER FOR not found set @res_table = 'EXISTS';

4.练习

——大家注意,存储过程的业务过程在java代码中一般也可以实现,我们下面的需求是为了练习存储过程

4.1 利用存储过程更新数据

为某部门(需指定)的人员涨薪100;如果是公司总裁,则不涨薪。

delimiter //create procedure high_sal(in dept_name varchar(32))begindeclare e_no int;declare e_name varchar(32);declare e_sal decimal(7,2);declare lp_flag boolean default true;declare emp_cursor cursor for select e.empno,e.ename,e.salfrom emp e,dept dwhere e.deptno = d.deptno and d.dname = dept_name;-- handler 句柄declare continue handler for NOT FOUND set lp_flag = false;open emp_cursor;emp_loop:loopfetch emp_cursor into e_no,e_name,e_sal;if lp_flag thenif e_name = 'king' then iterate emp_loop;else update emp e set e.sal = e.sal + 100 where e.empno = e_no;end if;elseleave emp_loop;end if;end loop emp_loop;set @end_falg = 'exit_flag';close emp_cursor;end //call high_sal('ACCOUNTING');

4.2 循环创建表

创建下个月的每天对应的表comp__04_01、comp__04_02、...(模拟)需求描述:我们需要用某个表记录很多数据,比如记录某某用户的搜索、购买行为(注意,此处是假设用数据库保存),当每天记录较多时,如果把所有数据都记录到一张表中太庞大,需要分表,我们的要求是,每天一张表,存当天的统计数据,就要求提前生产这些表——每月月底创建下一个月每天的表!

-- 知识点 预处理 prepare语句from后使用局部变量会报错 -- /doc/refman/5.6/en/sql-prepared-statements.htmlPREPARE stmt_name FROM preparable_stmtEXECUTE stmt_name [USING @var_name [, @var_name] ...]{DEALLOCATE | DROP} PREPARE stmt_name-- 知识点 时间的处理-- EXTRACT(unit FROM date)截取时间的指定位置值-- DATE_ADD(date,INTERVAL expr unit) 日期运算-- LAST_DAY(date) 获取日期的最后一天-- YEAR(date) 返回日期中的年-- MONTH(date) 返回日期的月-- DAYOFMONTH(date) 返回日

-- 思路:循环构建表名 comp__05_01 到 comp__05_31;并执行create语句。delimiter //create procedure sp_create_table()begindeclare next_year int;declare next_month int;declare next_month_day int;declare next_month_str char(2);declare next_month_day_str char(2);-- 处理每天的表名declare table_name_str char(10);declare t_index int default 1;-- declare create_table_sql varchar(200);-- 获取下个月的年份set next_year = year(date_add(now(),INTERVAL 1 month));-- 获取下个月是几月 set next_month = month(date_add(now(),INTERVAL 1 month));-- 下个月最后一天是几号set next_month_day = dayofmonth(LAST_DAY(date_add(now(),INTERVAL 1 month)));if next_month < 10then set next_month_str = concat('0',next_month);elseset next_month_str = concat('',next_month);end if;while t_index <= next_month_day doif (t_index < 10)then set next_month_day_str = concat('0',t_index);elseset next_month_day_str = concat('',t_index);end if;-- _05_01set table_name_str = concat(next_year,'_',next_month_str,'_',next_month_day_str);-- 拼接create sql语句set @create_table_sql = concat('create table comp_',table_name_str,'(`grade` INT(11) NULL,`losal` INT(11) NULL,`hisal` INT(11) NULL) COLLATE=\'utf8_general_ci\' ENGINE=InnoDB');-- FROM后面不能使用局部变量!prepare create_table_stmt FROM @create_table_sql;execute create_table_stmt;DEALLOCATE prepare create_table_stmt;set t_index = t_index + 1;end while;end//call sp_create_table()

4.3 其他场景

“为用户添加购物积分,并更新到用户的总积分表中”等需要对多张表进行CRUD操作的业务。而且内部可以使用事务命令。

5.其他

5.1 characteristic

characteristic:COMMENT 'string'| LANGUAGE SQL| [NOT] DETERMINISTIC| { CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA }| SQL SECURITY { DEFINER | INVOKER }

其中,SQL SECURITY的含义如下:

MySQL存储过程是通过指定SQL SECURITY子句指定执行存储过程的实际用户;如果SQL SECURITY子句指定为DEFINER(定义者),存储过程将使用存储过程的DEFINER执行存储过程,验证调用存储过程的用户是否具有存储过程的execute权限和DEFINER用户是否具有存储过程引用的相关对象(存储过程中的表等对象)的权限;如果SQL SECURITY子句指定为INVOKER(调用者),那么MySQL将使用当前调用存储过程的用户执行此过程,并验证用户是否具有存储过程的execute权限和存储过程引用的相关对象的权限;如果不显示的指定SQL SECURITY子句,MySQL默认将以DEFINER执行存储过程。

5.2 死循环处理

-- 如有死循环处理,可以通过下面的命令查看并结束show processlist;kill id;

5.3 可以在select语句中写case

/doc/refman/5.6/en/control-flow-functions.html

select casewhen timestampdiff(year,e.hiredate,now()) <= 38 then '新手'when timestampdiff(year,e.hiredate,now()) <= 40 then '老员工'else '元老'end hir_loc,e.*from emp e;

5.4 临时表

create temporary table 表名(字段名 类型 [约束],name varchar(20) )Engine=InnoDB default charset utf8;-- 需求:按照部门名称查询员工,通过select查看员工的编号、姓名、薪资。(注意,此处仅仅演示游标用法)delimiter $$create procedure sp_create_table02(in dept_name varchar(32))begindeclare emp_no int;declare emp_name varchar(32);declare emp_sal decimal(7,2);declare exit_flag int default 0;declare emp_cursor cursor forselect e.empno,e.ename,e.salfrom emp e inner join dept d on e.deptno = d.deptno where d.dname = dept_name;declare continue handler for not found set exit_flag = 1;-- 创建临时表收集数据CREATE temporary TABLE `temp_table_emp` (`empno` INT(11) NOT NULL COMMENT '员工编号',`ename` VARCHAR(32) NULL COMMENT '员工姓名' COLLATE 'utf8_general_ci',`sal` DECIMAL(7,2) NOT NULL DEFAULT '0.00' COMMENT '薪资',PRIMARY KEY (`empno`) USING BTREE)COLLATE='utf8_general_ci'ENGINE=InnoDB;open emp_cursor;c_loop:loopfetch emp_cursor into emp_no,emp_name,emp_sal;if exit_flag != 1 theninsert into temp_table_emp values(emp_no,emp_name,emp_sal); elseleave c_loop;end if;end loop c_loop;select * from temp_table_emp;select @sex_res; -- 仅仅是看一下会不会执行到close emp_cursor;end$$call sp_create_table02('RESEARCH');

5.5 复制表和数据

CREATE TABLE dept SELECT * FROM procedure_demo.dept;CREATE TABLE emp SELECT * FROM procedure_demo.emp;CREATE TABLE salgrade SELECT * FROM procedure_demo.salgrade;

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。