Redian新闻
>
mysql8.0流程控制一文拿捏

mysql8.0流程控制一文拿捏

公众号新闻

流程控制介绍

解决复杂问题不可能通过一个SQL语句完成,我们需要执行多个SQL操作。流程控制语句的作用就是控制存储过程中SQL语句的执行顺序,是我们完成复杂操作必不可少的一部分。只要是执行的程序,流程就分为三大类:   顺序结构:程序从上往下依次执行    分支结构:程序按条件进行选择执行,从两条或多条路径中选择一条执行    循环结构:程序满足一定条件下,重复执行一组语句 针对于MySQL的流程控制语句主要有3类。注意:只能用于存储程序。   条件判断语句:IF语句和CASE语句    循环语句:LOOP、WHILE和REPEAT语句    跳转语句:ITERATE和LEAVE语句

分支结构之IF

IF语句的语法结构是:IF 表达式1 THEN 操作1[ELSEIF 表达式2 THEN 操作2]……[ELSE 操作N]END IF根据表达式的结果为TRUE或FALSE执行相应的语句。这里“[]”中的内容是可选的。
特点:①不同的表达式对应不同的操作②使用在begin end中
举例1:DELIMITER //CREATE PROCEDURE test_if()BEGIN #情况1: #声明局部变量 #declare stu_name varchar(15); #if stu_name is null # then select 'stu_name is null'; #end if; #情况2:二选一 #declare email varchar(25) default 'aaa'; #if email is null # then select 'email is null'; #else # select 'email is not null'; #end if; #情况3:多选一 DECLARE age INT DEFAULT 20; IF age > 40 THEN SELECT '中老年'; ELSEIF age > 18 THEN SELECT '青壮年'; ELSEIF age > 8 THEN SELECT '青少年'; ELSE SELECT '婴幼儿'; END IF;END //DELIMITER ;
举例2:声明存储过程“update_salary_by_eid1”,定义IN参数emp_id,输入员工编号。判断该员工薪资如果低于8000元并且入职时间超过5年,就涨薪500元;否则就不变。DELIMITER //
CREATE PROCEDURE update_salary_by_eid1(IN emp_id INT)BEGIN #声明局部变量 DECLARE emp_sal DOUBLE; #记录员工的工资 DECLARE hire_year DOUBLE; #记录员工入职公司的年头 #赋值 SELECT salary INTO emp_sal FROM employees WHERE employee_id = emp_id; SELECT DATEDIFF(CURDATE(),hire_date)/365 INTO hire_year FROM employees WHERE employee_id = emp_id; #判断 IF emp_sal < 8000 AND hire_year >= 5 THEN UPDATE employees SET salary = salary + 500 WHERE employee_id = emp_id; END IF;END //DELIMITER ;#调用存储过程CALL update_salary_by_eid1(104);
SELECT DATEDIFF(CURDATE(),hire_date)/365, employee_id,salaryFROM employeesWHERE salary < 8000 AND DATEDIFF(CURDATE(),hire_date)/365 >= 5;
DROP PROCEDURE update_salary_by_eid1;
举例3:声明存储过程“update_salary_by_eid2”,定义IN参数emp_id,输入员工编号。判断该员工薪资如果低于9000元并且入职时间超过5年,就涨薪500元;否则就涨薪100元。DELIMITER //CREATE PROCEDURE update_salary_by_eid2(IN emp_id INT)BEGIN #声明局部变量 DECLARE emp_sal DOUBLE; #记录员工的工资 DECLARE hire_year DOUBLE; #记录员工入职公司的年头 #赋值 SELECT salary INTO emp_sal FROM employees WHERE employee_id = emp_id; SELECT DATEDIFF(CURDATE(),hire_date)/365 INTO hire_year FROM employees WHERE employee_id = emp_id; #判断 IF emp_sal < 9000 AND hire_year >= 5 THEN UPDATE employees SET salary = salary + 500 WHERE employee_id = emp_id; ELSE UPDATE employees SET salary = salary + 100 WHERE employee_id = emp_id; END IF;END //DELIMITER ;
#调用CALL update_salary_by_eid2(103);CALL update_salary_by_eid2(104);
SELECT * FROM employees WHERE employee_id IN (103,104);
#举例4:声明存储过程“update_salary_by_eid3”,定义IN参数emp_id,输入员工编号。#判断该员工薪资如果低于9000元,就更新薪资为9000元;薪资如果大于等于9000元且#低于10000的,但是奖金比例为NULL的,就更新奖金比例为0.01;其他的涨薪100元。DELIMITER //CREATE PROCEDURE update_salary_by_eid3(IN emp_id INT)BEGIN #声明变量 DECLARE emp_sal DOUBLE; #记录员工工资 DECLARE bonus DOUBLE; #记录员工的奖金率 #赋值 SELECT salary INTO emp_sal FROM employees WHERE employee_id = emp_id; SELECT commission_pct INTO bonus FROM employees WHERE employee_id = emp_id; #判断 IF emp_sal < 9000 THEN UPDATE employees SET salary = 9000 WHERE employee_id = emp_id; ELSEIF emp_sal < 10000 AND bonus IS NULL THEN UPDATE employees SET commission_pct = 0.01 WHERE employee_id = emp_id; ELSE UPDATE employees SET salary = salary + 100 WHERE employee_id = emp_id; END IF;END //DELIMITER ;


分支结构之CASE

CASE语句的语法结构1:#情况一:类似于switchCASE表达式WHEN 值1 THEN 结果1或语句1(如果是语句,需要加分号)WHEN 值2 THEN 结果2或语句2(如果是语句,需要加分号)...ELSE 结果n或语句n(如果是语句,需要加分号)END [case](如果是放在beginend中需要加上case,如果放在select后面不需要)
CASE语句的语法结构2:#情况二:类似于多重ifCASEWHEN 条件1 THEN 结果1或语句1(如果是语句,需要加分号)WHEN 条件2 THEN 结果2或语句2(如果是语句,需要加分号)...ELSE 结果n或语句n(如果是语句,需要加分号)END [case](如果是放在beginend中需要加上case,如果放在select后面不需要)
#举例1:基本使用DELIMITER //CREATE PROCEDURE test_case()BEGIN #演示1:case ... when ...then ... /* declare var int default 2; case var when 1 then select 'var = 1'; when 2 then select 'var = 2'; when 3 then select 'var = 3'; else select 'other value'; end case; */ #演示2:case when ... then .... DECLARE var1 INT DEFAULT 10; CASE WHEN var1 >= 100 THEN SELECT '三位数'; WHEN var1 >= 10 THEN SELECT '两位数'; ELSE SELECT '个数位'; END CASE;END //DELIMITER ;
#举例2:声明存储过程“update_salary_by_eid4”,定义IN参数emp_id,输入员工编号。#判断该员工薪资如果低于9000元,就更新薪资为9000元;薪资大于等于9000元且低于10000的,#但是奖金比例为NULL的,就更新奖金比例为0.01;其他的涨薪100元。DELIMITER //CREATE PROCEDURE update_salary_by_eid4(IN emp_id INT)BEGIN #局部变量的声明 DECLARE emp_sal DOUBLE; #记录员工的工资 DECLARE bonus DOUBLE; #记录员工的奖金率 #局部变量的赋值 SELECT salary INTO emp_sal FROM employees WHERE employee_id = emp_id; SELECT commission_pct INTO bonus FROM employees WHERE employee_id = emp_id; CASE WHEN emp_sal < 9000 THEN UPDATE employees SET salary = 9000 WHERE employee_id = emp_id; WHEN emp_sal < 10000 AND bonus IS NULL THEN UPDATE employees SET commission_pct = 0.01 WHERE employee_id = emp_id; ELSE UPDATE employees SET salary = salary + 100 WHERE employee_id = emp_id; END CASE;END //DELIMITER ;
#举例3:声明存储过程update_salary_by_eid5,定义IN参数emp_id,输入员工编号。#判断该员工的入职年限,如果是0年,薪资涨50;如果是1年,薪资涨100;#如果是2年,薪资涨200;如果是3年,薪资涨300;如果是4年,薪资涨400;其他的涨薪500。DELIMITER //CREATE PROCEDURE update_salary_by_eid5(IN emp_id INT)BEGIN #声明局部变量 DECLARE hire_year INT; #记录员工入职公司的总时间(单位:年) #赋值 SELECT ROUND(DATEDIFF(CURDATE(),hire_date) / 365) INTO hire_year FROM employees WHERE employee_id = emp_id; #判断 CASE hire_year WHEN 0 THEN UPDATE employees SET salary = salary + 50 WHERE employee_id = emp_id; WHEN 1 THEN UPDATE employees SET salary = salary + 100 WHERE employee_id = emp_id; WHEN 2 THEN UPDATE employees SET salary = salary + 200 WHERE employee_id = emp_id; WHEN 3 THEN UPDATE employees SET salary = salary + 300 WHERE employee_id = emp_id; WHEN 4 THEN UPDATE employees SET salary = salary + 400 WHERE employee_id = emp_id; ELSE UPDATE employees SET salary = salary + 500 WHERE employee_id = emp_id; END CASE;END //DELIMITER ;


循环结构之LOOP

LOOP循环语句用来重复执行某些语句。LOOP内的语句一直重复执行直到循环被退出(使用LEAVE子句),跳出循环过程。LOOP语句的基本格式如下:[loop_label:]LOOP循环执行的语句END LOOP [loop_label]其中,loop_label表示LOOP语句的标注名称,该参数可以省略。
#举例1:DELIMITER //CREATE PROCEDURE test_loop()BEGIN #声明局部变量 DECLARE num INT DEFAULT 1; loop_label:LOOP #重新赋值 SET num = num + 1; #可以考虑某个代码程序反复执行。(略) IF num >= 10 THEN LEAVE loop_label; END IF; END LOOP loop_label; #查看num SELECT num;END //DELIMITER ;
#举例2:当市场环境变好时,公司为了奖励大家,决定给大家涨工资。#声明存储过程“update_salary_loop()”,声明OUT参数num,输出循环次数。#存储过程中实现循环给大家涨薪,薪资涨为原来的1.1倍。直到全公司的平#均薪资达到12000结束。并统计循环次数。DELIMITER //CREATE PROCEDURE update_salary_loop(OUT num INT)BEGIN #声明变量 DECLARE avg_sal DOUBLE ; #记录员工的平均工资 DECLARE loop_count INT DEFAULT 0;#记录循环的次数 #① 初始化条件 #获取员工的平均工资 SELECT AVG(salary) INTO avg_sal FROM employees; loop_lab:LOOP #② 循环条件 #结束循环的条件 IF avg_sal >= 12000 THEN LEAVE loop_lab; END IF; #③ 循环体 #如果低于12000,更新员工的工资 UPDATE employees SET salary = salary * 1.1; #④ 迭代条件 #更新avg_sal变量的值 SELECT AVG(salary) INTO avg_sal FROM employees; #记录循环次数 SET loop_count = loop_count + 1; END LOOP loop_lab; #给num赋值 SET num = loop_count; END //DELIMITER ;
SELECT AVG(salary) FROM employees;CALL update_salary_loop(@num);SELECT @num;


循环结构之WHILE

WHILE语句创建一个带条件判断的循环过程。WHILE在执行语句执行时,先对指定的表达式进行判断,如果为真,就执行循环内的语句,否则退出循环。WHILE语句的基本格式如下:[while_label:] WHILE 循环条件  DO  循环体END WHILE [while_label];while_label为WHILE语句的标注名称;如果循环条件结果为真,WHILE语句内的语句或语句群被执行,直至循环条件为假,退出循环。
#举例1:DELIMITER //CREATE PROCEDURE test_while()BEGIN #初始化条件 DECLARE num INT DEFAULT 1; #循环条件 WHILE num <= 10 DO #循环体(略) #迭代条件 SET num = num + 1; END WHILE; #查询 SELECT num;END //DELIMITER ;
#调用CALL test_while();
#举例2:市场环境不好时,公司为了渡过难关,决定暂时降低大家的薪资。#声明存储过程“update_salary_while()”,声明OUT参数num,输出循环次数。#存储过程中实现循环给大家降薪,薪资降为原来的90%。直到全公司的平均薪资#达到5000结束。并统计循环次数。DELIMITER //CREATE PROCEDURE update_salary_while(OUT num INT)BEGIN #声明变量 DECLARE avg_sal DOUBLE ; #记录平均工资 DECLARE while_count INT DEFAULT 0; #记录循环次数 #赋值 SELECT AVG(salary) INTO avg_sal FROM employees; WHILE avg_sal > 5000 DO UPDATE employees SET salary = salary * 0.9 ; SET while_count = while_count + 1; SELECT AVG(salary) INTO avg_sal FROM employees; END WHILE; #给num赋值 SET num = while_count; END //DELIMITER ;
#调用CALL update_salary_while(@num);SELECT @num;SELECT AVG(salary) FROM employees;


循环结构之REPEAT

REPEAT语句创建一个带条件判断的循环过程。与WHILE循环不同的是,REPEAT循环首先会执行一次循环,然后在UNTIL中进行表达式的判断如果满足条件就退出,即ENDREPEAT;如果条件不满足,则会就继续执行循环,直到满足退出条件为止。REPEAT语句的基本格式如下:[repeat_label:] REPEAT    循环体的语句UNTIL 结束循环的条件表达式END REPEAT [repeat_label]repeat_label为REPEAT语句的标注名称,该参数可以省略;REPEAT语句内的语句或语句群被重复,直至expr_condition为真。
#举例1:DELIMITER //CREATE PROCEDURE test_repeat()BEGIN #声明变量 DECLARE num INT DEFAULT 1; REPEAT SET num = num + 1; UNTIL num >= 10 END REPEAT; #查看 SELECT num;END //DELIMITER ;
#调用CALL test_repeat();
#举例2:当市场环境变好时,公司为了奖励大家,决定给大家涨工资。#声明存储过程“update_salary_repeat()”,声明OUT参数num,输出循环次数。#存储过程中实现循环给大家涨薪,薪资涨为原来的1.15倍。直到全公司的平均#薪资达到13000结束。并统计循环次数。DELIMITER //CREATE PROCEDURE update_salary_repeat(OUT num INT)BEGIN #声明变量 DECLARE avg_sal DOUBLE ; #记录平均工资 DECLARE repeat_count INT DEFAULT 0; #记录循环次数 #赋值 SELECT AVG(salary) INTO avg_sal FROM employees; REPEAT UPDATE employees SET salary = salary * 1.15; SET repeat_count = repeat_count + 1; SELECT AVG(salary) INTO avg_sal FROM employees; UNTIL avg_sal >= 13000 END REPEAT; #给num赋值 SET num = repeat_count; END //DELIMITER ;
#调用CALL update_salary_repeat(@num);SELECT @num;SELECT AVG(salary) FROM employees;


对比三种循环结构

1、这三种循环都可以省略名称,但如果循环中添加了循环控制语句(LEAVE或ITERATE)则必须添加名称。2、LOOP:一般用于实现简单的"死"循环   WHILE:先判断后执行   REPEAT:先执行后判断,无条件至少执行一次


跳转语句之LEAVE

LEAVE语句:可以用在循环语句内,或者以BEGINEND包裹起来的程序体内,表示跳出循环或者跳出程序体的操作。如果你有面向过程的编程语言的使用经验,你可以把LEAVE理解为break。基本格式如下:LEAVE 标记名其中,label参数表示循环的标志。LEAVE和BEGIN...END或循环一起被使用。
举例1:创建存储过程“leave_begin()”,声明INT类型的IN参数num。给BEGIN...END加标记名,并在BEGIN...END中使用IF语句判断num参数的值。 如果num<=0,则使用LEAVE语句退出BEGIN...END; 如果num=1,则查询“employees”表的平均薪资; 如果num=2,则查询“employees”表的最低薪资; 如果num>2,则查询“employees”表的最高薪资。IF语句结束后查询“employees”表的总人数。DELIMITER //CREATE PROCEDURE leave_begin(IN num INT)begin_label:BEGIN IF num <= 0 THEN LEAVE begin_label; ELSEIF num = 1 THEN SELECT AVG(salary) FROM employees; ELSEIF num = 2 THEN SELECT MIN(salary) FROM employees; ELSE SELECT MAX(salary) FROM employees; END IF; #查询总人数 SELECT COUNT(*) FROM employees;END //DELIMITER ;
#调用CALL leave_begin(1);
#举例2:当市场环境不好时,公司为了渡过难关,决定暂时降低大家的薪资。#声明存储过程“leave_while()”,声明OUT参数num,输出循环次数,存储过程中使用WHILE#循环给大家降低薪资为原来薪资的90%,直到全公司的平均薪资小于等于10000,并统计循环次数。DELIMITER //CREATE PROCEDURE leave_while(OUT num INT)BEGIN DECLARE avg_sal DOUBLE;#记录平均工资 DECLARE while_count INT DEFAULT 0; #记录循环次数 SELECT AVG(salary) INTO avg_sal FROM employees; #① 初始化条件 while_label:WHILE TRUE DO #② 循环条件 #③ 循环体 IF avg_sal <= 10000 THEN LEAVE while_label; END IF; UPDATE employees SET salary = salary * 0.9; SET while_count = while_count + 1; #④ 迭代条件 SELECT AVG(salary) INTO avg_sal FROM employees; END WHILE; #赋值 SET num = while_count;END //DELIMITER ;
#调用CALL leave_while(@num);SELECT @num;
SELECT AVG(salary) FROM employees;


跳转语句之ITERATE

ITERATE语句:只能用在循环语句(LOOP、REPEAT和WHILE语句)内,表示重新开始循环,将执行顺序转到语句段开头处。如果你有面向过程的编程语言的使用经验,你可以把ITERATE理解为continue,意思为“再次循环”。语句基本格式如下:ITERATE labellabel参数表示循环的标志。ITERATE语句必须跟在循环标志前面。
举例:定义局部变量num,初始值为0。循环结构中执行num+1操作。 如果num<10,则继续执行循环; 如果num>15,则退出循环结构;DELIMITER //CREATE PROCEDURE test_iterate()BEGIN DECLARE num INT DEFAULT 0; loop_label:LOOP #赋值 SET num = num + 1; IF num < 10 THEN ITERATE loop_label; ELSEIF num > 15 THEN LEAVE loop_label; END IF; SELECT '尚硅谷:让天下没有难学的技术'; END LOOP;END //DELIMITER ;
CALL test_iterate();SELECT * FROM employees;


练习

#1. 创建函数test_if_case(),实现传入成绩,如果成绩>90,返回A,如果成绩>80,返回B,如果成绩>60,返回C,否则返回D#要求:分别使用if结构和case结构实现#方式1:DELIMITER //CREATE FUNCTION test_if_case1(score DOUBLE)RETURNS CHARBEGIN    DECLARE ch CHAR;    IF score>90        THEN SET ch='A';    ELSEIF score>80        THEN SET ch='B';    ELSEIF score>60        THEN SET ch='C';    ELSE SET ch='D';    END IF;    RETURN ch;END //DELIMITER ;
#调用SELECT test_if_case1(87);
#方式2:DELIMITER //CREATE FUNCTION test_if_case2(score DOUBLE)RETURNS CHARBEGIN DECLARE ch CHAR; CASE WHEN score>90 THEN SET ch='A'; WHEN score>80 THEN SET ch='B'; WHEN score>60 THEN SET ch='C'; ELSE SET ch='D'; END CASE; RETURN ch;END //DELIMITER ;
#调用SELECT test_if_case2(67);
#2. 创建存储过程test_if_pro(),传入工资值,如果工资值<3000,则删除工资为此值的员工,如果3000 <= 工资值 <= 5000,则修改此工资值的员工薪资涨1000,否则涨工资500DELIMITER //CREATE PROCEDURE test_if_pro(IN sal DOUBLE)BEGIN IF sal<3000 THEN DELETE FROM employees WHERE salary = sal; ELSEIF sal <= 5000 THEN UPDATE employees SET salary = salary+1000 WHERE salary = sal; ELSE UPDATE employees SET salary = salary+500 WHERE salary = sal; END IF;END //DELIMITER ;
SELECT * FROM employees;
#调用CALL test_if_pro(3100);
#3. 创建存储过程insert_data(),传入参数为 IN 的 INT 类型变量 insert_count,实现向admin表中批量插入insert_count条记录DELIMITER //CREATE PROCEDURE insert_data(IN insert_count INT)BEGIN DECLARE i INT DEFAULT 1; WHILE i <= insert_count DO INSERT INTO admin(user_name,user_pwd) VALUES(CONCAT('Rose-',i),ROUND(RAND() * 100000)); SET i=i+1; END WHILE;END //DELIMITER ;#调用CALL insert_data(100);

链接:https://blog.51cto.com/u_13236892/9127270

(版权归原作者所有,侵删)

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
MySQL如何性能调优?上篇美国的新对华政策还再用 Navicat?试试这款正版 MySQL 客户端,真香!谎言Mysql集群之PXC-Docker安装《幺妹和市场街》(3)Redis缓存与Mysql如何保证双写一致一文拿捏Redis7 发布订阅MySQL主从同步延迟原因与解决方案MySQL 支持 JavaScript,目前处于预览阶段ES+Redis+MySQL,这个高可用架构设计太顶了解密炒虾机器人远程控制技术:动捕手套/隔空取物/VR远程,都能训练机器人MYSQL事务的底层原理PostgreSQL数据脱敏方式盘点"是时候控制一下了"加拿大大幅削减学生签证数量一文解析 ODPS SQL 任务优化方法原理4 种 MySQL 同步 ES 方案,yyds!10 个完美替代 Navicat 的 MySQL 图形工具APEC还没结束,消失的8000流浪汉, 又双叒回来了..如何设计一款基于 MySQL 实现的 Message Queue纯爱小说﹕ 风云赋 (53) 门当户对MySQL 分库分表实践SQL Server死锁总结mysql8.0存储过程民國50年駐台美軍拍攝高雄街景的彩照Python如何使用MySQL 8.2读写分离?"是时候控制一下了"!加拿大大幅削减学生签证数量GitHub多项服务故障,与升级MySQL有关?基于 MySQL 多通道主主复制的机房容灾方案"是时候控制一下了"加拿大大幅削减学生签证数量!加拿大留学签证如何办理?MySQL 8.2 正式可用,支持读写分离1.8w 字详解 SQL 优化这些年背过的面试题——MySQL篇旧金山8000流浪汉一夜消失 特朗普:美国成印度了一文让你对mysql索引底层实现明明白白
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。