第18篇:流程控制语句介绍

star2017 1年前 ⋅ 480 阅读

Mysql系列的目标是:通过这个系列从入门到全面掌握一个高级开发所需要的全部技能。

欢迎大家加我微信itsoku一起交流java、算法、数据库相关技术。

这是Mysql系列第18篇。

环境:mysql5.7.25,cmd命令中进行演示。

代码中被[]包含的表示可选,|符号分开的表示可选其一。

上一篇存储过程&自定义函数,对存储过程和自定义函数做了一个简单的介绍,但是如何能够写出复杂的存储过程和函数呢?

这需要我们熟练掌握流程控制语句才可以,本文主要介绍mysql中流程控制语句的使用,上干货。

本篇内容

  • if函数
  • case语句
  • if结构
  • while循环
  • repeat循环
  • loop循环
  • 循环体控制语句

准备数据

  1. /*建库javacode2018*/
  2. drop database if exists javacode2018;
  3. create database javacode2018;
  4. /*切换到javacode2018库*/
  5. use javacode2018;
  6. /*创建表:t_user*/
  7. DROP TABLE IF EXISTS t_user;
  8. CREATE TABLE t_user(
  9. id int PRIMARY KEY COMMENT '编号',
  10. sex TINYINT not null DEFAULT 1 COMMENT '性别,1:男,2:女',
  11. name VARCHAR(16) not NULL DEFAULT '' COMMENT '姓名'
  12. )COMMENT '用户表';
  13. /*插入数据*/
  14. INSERT INTO t_user VALUES
  15. (1,1,'路人甲Java'),(2,1,'张学友'),(3,2,'王祖贤'),(4,1,'郭富城'),(5,2,'李嘉欣');
  16. SELECT * FROM t_user;
  17. DROP TABLE IF EXISTS test1;
  18. CREATE TABLE test1 (a int not null);
  19. DROP TABLE IF EXISTS test2;
  20. CREATE TABLE test2 (a int not null,b int NOT NULL );

if函数

语法

  1. if(条件表达式,值1,值2);

if函数有3个参数。

当参数1为true的时候,返回值1,否则返回值2

示例

需求:查询t_user表数据,返回:编号、性别(男、女)、姓名。

分析一下:数据库中性别用数字表示的,我们需要将其转换为(男、女),可以使用if函数。

  1. mysql> SELECT id 编号,if(sex=1,'男','女') 性别,name 姓名 FROM t_user;
  2. +--------+--------+---------------+
  3. | 编号 | 性别 | 姓名 |
  4. +--------+--------+---------------+
  5. | 1 | | 路人甲Java |
  6. | 2 | | 张学友 |
  7. | 3 | | 王祖贤 |
  8. | 4 | | 郭富城 |
  9. | 5 | | 李嘉欣 |
  10. +--------+--------+---------------+
  11. 5 rows in set (0.00 sec)

CASE结构

2种用法。

第1种用法

类似于java中的switch语句。

  1. case 表达式
  2. when 1 then 结果1或者语句1(如果是语句需要加分号)
  3. when 2 then 结果2或者语句2
  4. ...
  5. else 结果n或者语句n
  6. end [case] (如果是放在begin end之间需要加case,如果在select后则不需要)

示例1:select中使用

查询t_user表数据,返回:编号、性别(男、女)、姓名。

  1. /*写法1:类似于java中的if else*/
  2. SELECT id 编号,(CASE sex WHEN 1 THEN '男' ELSE '女' END) 性别,name 姓名 FROM t_user;
  3. /*写法2:类似于java中的if else if*/
  4. SELECT id 编号,(CASE sex WHEN 1 then '男' WHEN 2 then '女' END) 性别,name 姓名 FROM t_user;

示例2:begin end中使用

写一个存储过程,接受3个参数:id,性别(男、女),姓名,然后插入到t_user表

创建存储过程:

  1. /*删除存储过程proc1*/
  2. DROP PROCEDURE IF EXISTS proc1;
  3. /*s删除id=6的记录*/
  4. DELETE FROM t_user WHERE id=6;
  5. /*声明结束符为$*/
  6. DELIMITER $
  7. /*创建存储过程proc1*/
  8. CREATE PROCEDURE proc1(id int,sex_str varchar(8),name varchar(16))
  9. BEGIN
  10. /*声明变量v_sex用于存放性别*/
  11. DECLARE v_sex TINYINT UNSIGNED;
  12. /*根据sex_str的值来设置性别*/
  13. CASE sex_str
  14. when '男' THEN
  15. SET v_sex = 1;
  16. WHEN '女' THEN
  17. SET v_sex = 2;
  18. END CASE ;
  19. /*插入数据*/
  20. INSERT INTO t_user VALUES (id,v_sex,name);
  21. END $
  22. /*结束符置为;*/
  23. DELIMITER ;

调用存储过程:

  1. CALL proc1(6,'男','郭富城');

查看效果:

  1. mysql> select * from t_user;
  2. +----+-----+---------------+
  3. | id | sex | name |
  4. +----+-----+---------------+
  5. | 1 | 1 | 路人甲Java |
  6. | 2 | 1 | 张学友 |
  7. | 3 | 2 | 王祖贤 |
  8. | 4 | 1 | 郭富城 |
  9. | 5 | 2 | 李嘉欣 |
  10. | 6 | 1 | 郭富城 |
  11. +----+-----+---------------+
  12. 6 rows in set (0.00 sec)

示例3:函数中使用

需求:写一个函数,根据t_user表sex的值,返回男女

创建函数:

  1. /*删除存储过程proc1*/
  2. DROP FUNCTION IF EXISTS fun1;
  3. /*声明结束符为$*/
  4. DELIMITER $
  5. /*创建存储过程proc1*/
  6. CREATE FUNCTION fun1(sex TINYINT UNSIGNED)
  7. RETURNS varchar(8)
  8. BEGIN
  9. /*声明变量v_sex用于存放性别*/
  10. DECLARE v_sex VARCHAR(8);
  11. CASE sex
  12. WHEN 1 THEN
  13. SET v_sex:='男';
  14. ELSE
  15. SET v_sex:='女';
  16. END CASE;
  17. RETURN v_sex;
  18. END $
  19. /*结束符置为;*/
  20. DELIMITER ;

看一下效果:

  1. mysql> select sex, fun1(sex) 性别,name FROM t_user;
  2. +-----+--------+---------------+
  3. | sex | 性别 | name |
  4. +-----+--------+---------------+
  5. | 1 | | 路人甲Java |
  6. | 1 | | 张学友 |
  7. | 2 | | 王祖贤 |
  8. | 1 | | 郭富城 |
  9. | 2 | | 李嘉欣 |
  10. | 1 | | 郭富城 |
  11. +-----+--------+---------------+
  12. 6 rows in set (0.00 sec)

第2种用法

类似于java中多重if语句。

  1. case
  2. when 条件1 then 结果1或者语句1(如果是语句需要加分号)
  3. when 条件2 then 结果2或者语句2
  4. ...
  5. else 结果n或者语句n
  6. end [case] (如果是放在begin end之间需要加case,如果是在select后面case可以省略)

这种写法和1中的类似,大家用上面这种语法实现第1中用法中的3个示例,贴在留言中。

if结构

if结构类似于java中的 if..else if…else的语法,如下:

  1. if 条件语句1 then 语句1;
  2. elseif 条件语句2 then 语句2;
  3. ...
  4. else 语句n;
  5. end if;

只能使用在begin end之间。

示例

写一个存储过程,实现用户数据的插入和新增,如果id存在,则修改,不存在则新增,并返回结果

  1. /*删除id=7的记录*/
  2. DELETE FROM t_user WHERE id=7;
  3. /*删除存储过程*/
  4. DROP PROCEDURE IF EXISTS proc2;
  5. /*声明结束符为$*/
  6. DELIMITER $
  7. /*创建存储过程*/
  8. CREATE PROCEDURE proc2(v_id int,v_sex varchar(8),v_name varchar(16),OUT result TINYINT)
  9. BEGIN
  10. DECLARE v_count TINYINT DEFAULT 0;/*用来保存user记录的数量*/
  11. /*根据v_id查询数据放入v_count中*/
  12. select count(id) into v_count from t_user where id = v_id;
  13. /*v_count>0表示数据存在,则修改,否则新增*/
  14. if v_count>0 THEN
  15. BEGIN
  16. DECLARE lsex TINYINT;
  17. select if(lsex='男',1,2) into lsex;
  18. update t_user set sex = lsex,name = v_name where id = v_id;
  19. /*获取update影响行数*/
  20. select ROW_COUNT() INTO result;
  21. END;
  22. else
  23. BEGIN
  24. DECLARE lsex TINYINT;
  25. select if(lsex='男',1,2) into lsex;
  26. insert into t_user VALUES (v_id,lsex,v_name);
  27. select 0 into result;
  28. END;
  29. END IF;
  30. END $
  31. /*结束符置为;*/
  32. DELIMITER ;

看效果:

  1. mysql> SELECT * FROM t_user;
  2. +----+-----+---------------+
  3. | id | sex | name |
  4. +----+-----+---------------+
  5. | 1 | 1 | 路人甲Java |
  6. | 2 | 1 | 张学友 |
  7. | 3 | 2 | 王祖贤 |
  8. | 4 | 1 | 郭富城 |
  9. | 5 | 2 | 李嘉欣 |
  10. | 6 | 1 | 郭富城 |
  11. +----+-----+---------------+
  12. 6 rows in set (0.00 sec)
  13. mysql> CALL proc2(7,'男','黎明',@result);
  14. Query OK, 1 row affected (0.00 sec)
  15. mysql> SELECT @result;
  16. +---------+
  17. | @result |
  18. +---------+
  19. | 0 |
  20. +---------+
  21. 1 row in set (0.00 sec)
  22. mysql> SELECT * FROM t_user;
  23. +----+-----+---------------+
  24. | id | sex | name |
  25. +----+-----+---------------+
  26. | 1 | 1 | 路人甲Java |
  27. | 2 | 1 | 张学友 |
  28. | 3 | 2 | 王祖贤 |
  29. | 4 | 1 | 郭富城 |
  30. | 5 | 2 | 李嘉欣 |
  31. | 6 | 1 | 郭富城 |
  32. | 7 | 2 | 黎明 |
  33. +----+-----+---------------+
  34. 7 rows in set (0.00 sec)
  35. mysql> CALL proc2(7,'男','梁朝伟',@result);
  36. Query OK, 1 row affected (0.00 sec)
  37. mysql> SELECT @result;
  38. +---------+
  39. | @result |
  40. +---------+
  41. | 1 |
  42. +---------+
  43. 1 row in set (0.00 sec)
  44. mysql> SELECT * FROM t_user;
  45. +----+-----+---------------+
  46. | id | sex | name |
  47. +----+-----+---------------+
  48. | 1 | 1 | 路人甲Java |
  49. | 2 | 1 | 张学友 |
  50. | 3 | 2 | 王祖贤 |
  51. | 4 | 1 | 郭富城 |
  52. | 5 | 2 | 李嘉欣 |
  53. | 6 | 1 | 郭富城 |
  54. | 7 | 2 | 梁朝伟 |
  55. +----+-----+---------------+
  56. 7 rows in set (0.00 sec)

循环

mysql中循环有3种写法

  1. while:类似于java中的while循环
  2. repeat:类似于java中的do while循环
  3. loop:类似于java中的while(true)死循环,需要在内部进行控制。

循环控制

对循环内部的流程进行控制,如:

结束本次循环

类似于java中的continue

  1. iterate 循环标签;

退出循环

类似于java中的break

  1. leave 循环标签;

下面我们分别介绍3种循环的使用。

while循环

类似于java中的while循环。

语法

  1. [标签:]while 循环条件 do
  2. 循环体
  3. end while [标签];

标签:是给while取个名字,标签和iterateleave结合用于在循环内部对循环进行控制:如:跳出循环、结束本次循环。

注意:这个循环先判断条件,条件成立之后,才会执行循环体,每次执行都会先进行判断。

示例1:无循环控制语句

根据传入的参数v_count向test1表插入指定数量的数据。

  1. /*删除test1表记录*/
  2. DELETE FROM test1;
  3. /*删除存储过程*/
  4. DROP PROCEDURE IF EXISTS proc3;
  5. /*声明结束符为$*/
  6. DELIMITER $
  7. /*创建存储过程*/
  8. CREATE PROCEDURE proc3(v_count int)
  9. BEGIN
  10. DECLARE i int DEFAULT 1;
  11. a:WHILE i<=v_count DO
  12. INSERT into test1 values (i);
  13. SET i=i+1;
  14. END WHILE;
  15. END $
  16. /*结束符置为;*/
  17. DELIMITER ;

见效果:

  1. mysql> CALL proc3(5);
  2. Query OK, 1 row affected (0.01 sec)
  3. mysql> SELECT * from test1;
  4. +---+
  5. | a |
  6. +---+
  7. | 1 |
  8. | 2 |
  9. | 3 |
  10. | 4 |
  11. | 5 |
  12. +---+
  13. 5 rows in set (0.00 sec)

示例2:添加leave控制语句

根据传入的参数v_count向test1表插入指定数量的数据,当插入超过10条,结束。

  1. /*删除存储过程*/
  2. DROP PROCEDURE IF EXISTS proc4;
  3. /*声明结束符为$*/
  4. DELIMITER $
  5. /*创建存储过程*/
  6. CREATE PROCEDURE proc4(v_count int)
  7. BEGIN
  8. DECLARE i int DEFAULT 1;
  9. a:WHILE i<=v_count DO
  10. INSERT into test1 values (i);
  11. /*判断i=10,离开循环a*/
  12. IF i=10 THEN
  13. LEAVE a;
  14. END IF;
  15. SET i=i+1;
  16. END WHILE;
  17. END $
  18. /*结束符置为;*/
  19. DELIMITER ;

见效果:

  1. mysql> DELETE FROM test1;
  2. Query OK, 20 rows affected (0.00 sec)
  3. mysql> CALL proc4(20);
  4. Query OK, 1 row affected (0.02 sec)
  5. mysql> SELECT * from test1;
  6. +----+
  7. | a |
  8. +----+
  9. | 1 |
  10. | 2 |
  11. | 3 |
  12. | 4 |
  13. | 5 |
  14. | 6 |
  15. | 7 |
  16. | 8 |
  17. | 9 |
  18. | 10 |
  19. +----+
  20. 10 rows in set (0.00 sec)

示例3:添加iterate控制语句

根据传入的参数v_count向test1表插入指定数量的数据,只插入偶数数据。

  1. /*删除test1表记录*/
  2. DELETE FROM test1;
  3. /*删除存储过程*/
  4. DROP PROCEDURE IF EXISTS proc5;
  5. /*声明结束符为$*/
  6. DELIMITER $
  7. /*创建存储过程*/
  8. CREATE PROCEDURE proc5(v_count int)
  9. BEGIN
  10. DECLARE i int DEFAULT 0;
  11. a:WHILE i<=v_count DO
  12. SET i=i+1;
  13. /*如果i不为偶数,跳过本次循环*/
  14. IF i%2!=0 THEN
  15. ITERATE a;
  16. END IF;
  17. /*插入数据*/
  18. INSERT into test1 values (i);
  19. END WHILE;
  20. END $
  21. /*结束符置为;*/
  22. DELIMITER ;

见效果:

  1. mysql> DELETE FROM test1;
  2. Query OK, 5 rows affected (0.00 sec)
  3. mysql> CALL proc5(10);
  4. Query OK, 1 row affected (0.01 sec)
  5. mysql> SELECT * from test1;
  6. +----+
  7. | a |
  8. +----+
  9. | 2 |
  10. | 4 |
  11. | 6 |
  12. | 8 |
  13. | 10 |
  14. +----+
  15. 5 rows in set (0.00 sec)

示例4:嵌套循环

test2表有2个字段(a,b),写一个存储过程(2个参数:v_a_count,v_b_count),使用双重循环插入数据,数据条件:a的范围[1,v_a_count]、b的范围[1,v_b_count]所有偶数的组合。

  1. /*删除存储过程*/
  2. DROP PROCEDURE IF EXISTS proc8;
  3. /*声明结束符为$*/
  4. DELIMITER $
  5. /*创建存储过程*/
  6. CREATE PROCEDURE proc8(v_a_count int,v_b_count int)
  7. BEGIN
  8. DECLARE v_a int DEFAULT 0;
  9. DECLARE v_b int DEFAULT 0;
  10. a:WHILE v_a<=v_a_count DO
  11. SET v_a=v_a+1;
  12. SET v_b=0;
  13. b:WHILE v_b<=v_b_count DO
  14. SET v_b=v_b+1;
  15. IF v_a%2!=0 THEN
  16. ITERATE a;
  17. END IF;
  18. IF v_b%2!=0 THEN
  19. ITERATE b;
  20. END IF;
  21. INSERT INTO test2 VALUES (v_a,v_b);
  22. END WHILE b;
  23. END WHILE a;
  24. END $
  25. /*结束符置为;*/
  26. DELIMITER ;

代码中故意将ITERATE a;放在内层循环中,主要让大家看一下效果。

见效果:

  1. mysql> DELETE FROM test2;
  2. Query OK, 6 rows affected (0.00 sec)
  3. mysql> CALL proc8(4,6);
  4. Query OK, 1 row affected (0.01 sec)
  5. mysql> SELECT * from test2;
  6. +---+---+
  7. | a | b |
  8. +---+---+
  9. | 2 | 2 |
  10. | 2 | 4 |
  11. | 2 | 6 |
  12. | 4 | 2 |
  13. | 4 | 4 |
  14. | 4 | 6 |
  15. +---+---+
  16. 6 rows in set (0.00 sec)

repeat循环

语法

  1. [标签:]repeat
  2. 循环体;
  3. until 结束循环的条件 end repeat [标签];

repeat循环类似于java中的do…while循环,不管如何,循环都会先执行一次,然后再判断结束循环的条件,不满足结束条件,循环体继续执行。这块和while不同,while是先判断条件是否成立再执行循环体。

示例1:无循环控制语句

根据传入的参数v_count向test1表插入指定数量的数据。

  1. /*删除存储过程*/
  2. DROP PROCEDURE IF EXISTS proc6;
  3. /*声明结束符为$*/
  4. DELIMITER $
  5. /*创建存储过程*/
  6. CREATE PROCEDURE proc6(v_count int)
  7. BEGIN
  8. DECLARE i int DEFAULT 1;
  9. a:REPEAT
  10. INSERT into test1 values (i);
  11. SET i=i+1;
  12. UNTIL i>v_count END REPEAT;
  13. END $
  14. /*结束符置为;*/
  15. DELIMITER ;

见效果:

  1. mysql> DELETE FROM test1;
  2. Query OK, 1 row affected (0.00 sec)
  3. mysql> CALL proc6(5);
  4. Query OK, 1 row affected (0.01 sec)
  5. mysql> SELECT * from test1;
  6. +---+
  7. | a |
  8. +---+
  9. | 1 |
  10. | 2 |
  11. | 3 |
  12. | 4 |
  13. | 5 |
  14. +---+
  15. 5 rows in set (0.00 sec)

repeat中iterateleave用法和while中类似,这块的示例算是给大家留的作业,写好的发在留言区,谢谢。

loop循环

语法

  1. [标签:]loop
  2. 循环体;
  3. end loop [标签];

loop相当于一个死循环,需要在循环体中使用iterate或者leave来控制循环的执行。

示例1:无循环控制语句

根据传入的参数v_count向test1表插入指定数量的数据。

  1. /*删除存储过程*/
  2. DROP PROCEDURE IF EXISTS proc7;
  3. /*声明结束符为$*/
  4. DELIMITER $
  5. /*创建存储过程*/
  6. CREATE PROCEDURE proc7(v_count int)
  7. BEGIN
  8. DECLARE i int DEFAULT 0;
  9. a:LOOP
  10. SET i=i+1;
  11. /*当i>v_count的时候退出循环*/
  12. IF i>v_count THEN
  13. LEAVE a;
  14. END IF;
  15. INSERT into test1 values (i);
  16. END LOOP a;
  17. END $
  18. /*结束符置为;*/
  19. DELIMITER ;

见效果:

  1. mysql> DELETE FROM test1;
  2. Query OK, 5 rows affected (0.00 sec)
  3. mysql> CALL proc7(5);
  4. Query OK, 1 row affected (0.01 sec)
  5. mysql> SELECT * from test1;
  6. +---+
  7. | a |
  8. +---+
  9. | 1 |
  10. | 2 |
  11. | 3 |
  12. | 4 |
  13. | 5 |
  14. +---+
  15. 5 rows in set (0.00 sec)

loop中iterateleave用法和while中类似,这块的示例算是给大家留的作业,写好的发在留言区,谢谢。

总结

  1. 本文主要介绍了mysql中控制流语句的使用,请大家下去了多练习,熟练掌握
  2. if函数常用在select中
  3. case语句有2种写法,主要用在select、begin end中,select中end后面可以省略case,begin end中使用不能省略case
  4. if语句用在begin end中
  5. 3种循环体的使用,while类似于java中的while循环,repeat类似于java中的do while循环,loop类似于java中的死循环,都用于begin end中
  6. 循环中体中的控制依靠leaveiterateleave类似于java中的break可以退出循环,iterate类似于java中的continue可以结束本次循环

最新资料

更多内容请访问:IT源点

相关文章推荐

全部评论: 0

    我有话说: