玩转Mysql系列 - 第18篇:流程控制语句(高手进阶)

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函数

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

调用存储过程:

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  OK1 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  OK1 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

iterate 循环标签;
退出循环

类似于java中的break

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  OK1 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  OK20 rows affected ( 0.00 sec)
  3. mysql>  CALL  proc4( 20);
  4. Query  OK1 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  OK5 rows affected ( 0.00 sec)
  3. mysql>  CALL  proc5( 10);
  4. Query  OK1 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  OK6 rows affected ( 0.00 sec)
  3. mysql>  CALL  proc8( 4, 6);
  4. Query  OK1 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  OK1 row affected ( 0.00 sec)
  3. mysql>  CALL  proc6( 5);
  4. Query  OK1 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  OK5 rows affected ( 0.00 sec)
  3. mysql>  CALL  proc7( 5);
  4. Query  OK1 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可以结束本次循环

Mysql系列目录

  1. 第1篇:mysql基础知识

  2. 第2篇:详解mysql数据类型(重点)

  3. 第3篇:管理员必备技能(必须掌握)

  4. 第4篇:DDL常见操作

  5. 第5篇:DML操作汇总(insert,update,delete)

  6. 第6篇:select查询基础篇

  7. 第7篇:玩转select条件查询,避免采坑

  8. 第8篇:详解排序和分页(order by & limit)

  9. 第9篇:分组查询详解(group by & having)

  10. 第10篇:常用的几十个函数详解

  11. 第11篇:深入了解连接查询及原理

  12. 第12篇:子查询

  13. 第13篇:细说NULL导致的神坑,让人防不胜防

  14. 第14篇:详解事务

  15. 第15篇:详解视图

  16. 第16篇:变量详解

  17. 第17篇:存储过程&自定义函数详解

mysql系列大概有20多篇,喜欢的请关注一下,欢迎大家加我微信itsoku或者留言交流mysql相关技术!

路人甲java

640?wx_fmt=png

▲长按图片识别二维码关注

路人甲java:工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活

来源:https://itsoku.blog.csdn.net/article/details/102383358
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值