背景

说到流程控制语句,我们在程序语法中用的比较多,比如C#的if..else...,while...,?: 等。同样的,在MySQL中,也有一些流程控制的语法,方便我们在写函数、存储过程的时候对逻辑进行控制和处理。

常见的过程式SQL语句可以用在存储过程或者函数体中。其中包括:IF函数、IF条件语句、CASE语句、LOOP语句、WHILE语句、REPEAT语句、LEAVE语句和ITERATE语句,它们极大的方便了我们进行流程控制。

下面我们一个一个来看。

流程语句分解

数据基础

 1 mysql> select * from students;
2 +-----------+-------------+-------+---------+-----+
3 | studentid | studentname | score | classid | sex |
4 +-----------+-------------+-------+---------+-----+
5 | 1 | brand | 105.5 | 1 | 1 |
6 | 2 | helen | 98.5 | 1 | 0 |
7 | 3 | lyn | 97 | 1 | 0 |
8 | 4 | sol | 97 | 1 | 1 |
9 | 5 | b1 | 89 | 2 | 1 |
10 | 6 | b2 | 90 | 2 | 1 |
11 | 7 | c1 | 76 | 3 | 0 |
12 | 8 | c2 | 73.5 | 3 | 0 |
13 | 9 | lala | 73 | 0 | 0 |
14 | 10 | A | 100 | 3 | 1 |
15 | 16 | test1 | 100 | 0 | 1 |
16 | 17 | trigger2 | 107 | 0 | 1 |
17 | 22 | trigger1 | 100 | 0 | 0 |
18 +-----------+-------------+-------+---------+-----+
19 13 rows in set
20
21 mysql> select * from scores;
22 +-----------+---------+-------+
23 | scoregrad | downset | upset |
24 +-----------+---------+-------+
25 | A | 81 | 90 |
26 | B | 71 | 80 |
27 | C | 61 | 70 |
28 | D | 51 | 60 |
29 | S | 91 | 100 |
30 | S+ | 101 | 120 |
31 +-----------+---------+-------+
32 6 rows in set 

IF函数

有点类似C#语法中的三元表达式,有3个参数,第一个参数是表达式,后面两个是值,当表达式成立的时候取第一个值,表达式不成立的时候取第二个值。

1 if(expr,val1,val2); --语法 

输出学生信息中的名称和性别(1为男,0为女,这边用if函数进行转换)

 1 mysql> select studentname,if(sex=0,'女','男') from students where classid<>0;
2 +-------------+---------------------+
3 | studentname | if(sex=0,'女','男') |
4 +-------------+---------------------+
5 | brand | 男 |
6 | helen | 女 |
7 | lyn | 女 |
8 | sol | 男 |
9 | b1 | 男 |
10 | b2 | 男 |
11 | c1 | 女 |
12 | c2 | 女 |
13 | A | 男 |
14 +-------------+---------------------+
15 9 rows in set 

IF条件语句

IF语句用来进行条件判断,根据不同的条件执行不同的操作。该语句在执行时首先判断IF后的条件是否为真,则执行THEN后的语句,如果为假则继续判断IF语句直到为真为止,当以上都不满足时则执行ELSE语句后的内容。

1 IF condition THEN
2 ...
3 ELSEIF condition THEN
4 ...
5 ELSE
6 ...
7 END IF 

代码示例,根据考试成绩来分布不同的成绩等级

 1 mysql>
2 /*如果存在函数func_test2,则删除*/
3 DROP FUNCTION IF EXISTS fun_if;
4 /*声明结束符为$*/
5 DELIMITER $
6 /*创建函数*/
7 CREATE FUNCTION fun_if(score DECIMAL(10,2))
8 RETURNS CHAR
9 BEGIN
10 DECLARE score_grad VARCHAR(5) DEFAULT '';
11 IF score>100 THEN SET score_grad='S';
12 ELSEIF (score BETWEEN 91 AND 100) THEN SET score_grad='A';
13 ELSEIF (score BETWEEN 81 AND 90) THEN SET score_grad='B';
14 ELSEIF (score BETWEEN 71 AND 80) THEN SET score_grad='C' ;
15 ELSE set score_grad='D';
16 END IF;
17 return score_grad;
18 END $
19 /*重置结束符为;*/
20 DELIMITER ;
21 Query OK, 0 rows affected 

执行结果

1 mysql> select fun_if(101),fun_if(100),fun_if(90),fun_if(80),fun_if(70);
2 +-------------+-------------+------------+------------+------------+
3 | fun_if(101) | fun_if(100) | fun_if(90) | fun_if(80) | fun_if(70) |
4 +-------------+-------------+------------+------------+------------+
5 | S | A | B | C | D |
6 +-------------+-------------+------------+------------+------------+
7 1 row in set 

CASE语句

CASE语句为多分支语句结构,该语句首先从WHEN后的VALUE中查找与CASE后的VALUE相等的值,如果查找到则执行该分支的内容,否则执行ELSE后的内容。CASE语句表示形式如下,类似C#中switch:

1 CASE expr
2 WHEN val1 THEN result1 or state1[;](可选项,如果是语句需要加分号,结果值可以加)
3 WHEN val2 THEN result2 or state2
4 ...
5 ELSE resultn or staten
6 END [CASE] (可选项,在begin end之间需加case,select后就不需要) 

在select中使用示例

 1 mysql> select studentname,case sex WHEN 0 THEN '女' WHEN 1 THEN '男' end as sex
2 from students where classid<>0;
3 +-------------+-----+
4 | studentname | sex |
5 +-------------+-----+
6 | brand | 男 |
7 | helen | 女 |
8 | lyn | 女 |
9 | sol | 男 |
10 | b1 | 男 |
11 | b2 | 男 |
12 | c1 | 女 |
13 | c2 | 女 |
14 | A | 男 |
15 +-------------+-----+
16 9 rows in set 

在函数或存储过程中使用示例

 1 mysql>
2 /*如果存在函数func_test2,则删除*/
3 DROP FUNCTION IF EXISTS fun_case;
4 /*声明结束符为$*/
5 DELIMITER $
6 /*创建函数*/
7 CREATE FUNCTION fun_case(sex INT)
8 RETURNS VARCHAR(20)
9 BEGIN
10 DECLARE sexStr VARCHAR(20) DEFAULT '';
11 CASE sex
12 WHEN 0 then set sexStr='女';
13 WHEN 1 then set sexStr='男';
14 ELSE set sexStr='不确定';
15 END CASE;
16 return sexStr;
17 END $
18 /*重置结束符为;*/
19 DELIMITER ;
20
21 Query OK, 0 rows affected 

函数执行结果

 1 mysql> select studentname,fun_case(sex) from students where classid<>0;
2 +-------------+---------------+
3 | studentname | fun_case(sex) |
4 +-------------+---------------+
5 | brand | 男 |
6 | helen | 女 |
7 | lyn | 女 |
8 | sol | 男 |
9 | b1 | 男 |
10 | b2 | 男 |
11 | c1 | 女 |
12 | c2 | 女 |
13 | A | 男 |
14 +-------------+---------------+
15 9 rows in set 

循环语句while

循环语句while 类似于C#中的while循环,我们知道在C#的while 或者 for 语句中,经常有用到两个关键语法:跳过当前循环(continue) 和 结束循环(break)。

同样的,在MySQL中也有两个语法对应跳过和结束循环。

1 ITERATE loop_label; --跳过当前循环
1 LEAVE loop_label; --结束循环
while 语法
1 [loop_label:]while condition do
2 --Todo:loop body
3 end while [loop_label]; 

loop_label:循环标签,和iterateleave结合用于在循环内部对循环进行控制:如:跳过本次循环、结束循环。

condition:循环条件,当满足条件的时候,就会执行循环体,条件不成立的时候结束循环。

while示例

下面脚本代码演示了将students表中studentid在给定数值范围内的数据存储到另外一张表中。

 1 /*先清除studentCount表记录*/
2 truncate table studentcount;
3 /*存储过程如果存在则删除*/
4 DROP PROCEDURE IF EXISTS sp_while1;
5 /*声明结束符为$*/
6 DELIMITER $
7 /*创建存储过程*/
8 CREATE PROCEDURE sp_while1(varial_count int)
9 BEGIN
10 DECLARE idx int DEFAULT 1;
11 DECLARE uname VARCHAR(30) DEFAULT '';
12 loop_label:WHILE idx<=varial_count DO
13 select studentname into uname from students where studentid = idx;
14 INSERT into studentCount values (idx,uname);
15 SET idx=idx+1;
16 END WHILE;
17 END $
18 /*结束符置为;*/
19 DELIMITER ; 

调用存储过程,给定数值范围是10,所以这边取出1~10的数据存储到studentCount表中

 1 mysql> CALL sp_while1(10);
2 Query OK, 1 row affected
3
4 mysql> select * from studentCount;
5 +-----------+-------------+
6 | studentid | studentname |
7 +-----------+-------------+
8 | 1 | brand |
9 | 2 | helen |
10 | 3 | lyn |
11 | 4 | sol |
12 | 5 | b1 |
13 | 6 | b2 |
14 | 7 | c1 |
15 | 8 | c2 |
16 | 9 | lala |
17 | 10 | A |
18 +-----------+-------------+
19 10 rows in set 
while示例:包含iterate/leave

前面我们说明过了,iterate 和 leave 分别代表跳过本次循环,类似于C#中的continue和break。我们在例子中测试下吧:

遇到studentname=lala时,结束循环,遇到偶数时候跳过单次循环。

 1 /*先清除studentCount表记录*/
2 truncate table studentcount;
3 /*存储过程如果存在则删除*/
4 DROP PROCEDURE IF EXISTS sp_while2;
5 /*声明结束符为$*/
6 DELIMITER $
7 /*创建存储过程*/
8 CREATE PROCEDURE sp_while2(varial_count int)
9 BEGIN
10 DECLARE idx int DEFAULT 0;
11 DECLARE uname VARCHAR(30) DEFAULT '';
12 loop_label:WHILE idx<=varial_count DO
13 SET idx=idx+1;
14 select studentname into uname from students where studentid = idx;
15 /*如果遇到studentname为lala的同学,结束循环*/
16 IF uname='lala' THEN
17 LEAVE loop_label;
18 /*如果idx为偶数,则跳过本次循环*/
19 ELSEIF idx%2=0 THEN
20 ITERATE loop_label;
21 END IF;
22 INSERT into studentCount values (idx,uname);
23 END WHILE;
24 END $
25 /*结束符置为;*/
26 DELIMITER ; 

调用存储过程,输出符合要求的数据:

 1 mysql> CALL sp_while2(10);
2 Query OK, 1 row affected
3
4 mysql> select * from studentCount;
5 +-----------+-------------+
6 | studentid | studentname |
7 +-----------+-------------+
8 | 1 | brand |
9 | 3 | lyn |
10 | 5 | b1 |
11 | 7 | c1 |
12 +-----------+-------------+
13 4 rows in set

循环语句repeat

repeat语法
1 [loop_label:]repeat
2 -- Todo loop body
3 until condition
4 end repeat [loop_label]; 

可以对比下上面while的语法,while是先判断条件是否成立再执行循环体,repeat循环更像是的do...while循环,就是循环始终都会先执行一次,然后再判断结束循环的条件,不满足结束条件,循环体继续执行。

 1 /*先清除studentCount表记录*/
2 truncate table studentcount;
3 /*存储过程如果存在则删除*/
4 DROP PROCEDURE IF EXISTS sp_repeat;
5 /*声明结束符为$*/
6 DELIMITER $
7 /*创建存储过程*/
8 CREATE PROCEDURE sp_repeat(varial_count int)
9 BEGIN
10 DECLARE idx int DEFAULT 0;
11 DECLARE uname VARCHAR(30) DEFAULT '';
12 loop_label:REPEAT
13 SET idx=idx+1;
14 select studentname into uname from students where studentid = idx;
15 /*如果遇到studentname为lala的同学,结束循环*/
16 IF uname='lala' THEN
17 LEAVE loop_label;
18 /*如果idx为偶数,则跳过本次循环*/
19 ELSEIF idx%2=0 THEN
20 ITERATE loop_label;
21 END IF;
22 INSERT into studentCount values (idx,uname);
23 UNTIL idx>varial_count
24 END REPEAT;
25 END $
26 /*结束符置为;*/
27 DELIMITER ; 

注意条件的变化,下面是调用存储过程,输出需要的数据:

 1 mysql> CALL sp_repeat(10);
2 Query OK, 1 row affected
3
4 mysql> select * from studentCount;
5 +-----------+-------------+
6 | studentid | studentname |
7 +-----------+-------------+
8 | 1 | brand |
9 | 3 | lyn |
10 | 5 | b1 |
11 | 7 | c1 |
12 +-----------+-------------+
13 4 rows in set 

循环语句loop

loop语法
1 [loop_label:]loop
2 --Todo loop body
3 end loop [loop label]; 

loop不像while和repeat那样有控制条件,条件不符合的时候会跳出。所以它实际上是会一直执行的,如果不主动中断或者跳出的话,类似于一个死循环,需要在循环体中使用iterate或者leave来控制循环的执行。

 1 /*先清除studentCount表记录*/
2 truncate table studentcount;
3 /*存储过程如果存在则删除*/
4 DROP PROCEDURE IF EXISTS sp_loop;
5 /*声明结束符为$*/
6 DELIMITER $
7 /*创建存储过程*/
8 CREATE PROCEDURE sp_loop(varial_count int)
9 BEGIN
10 DECLARE idx int DEFAULT 0;
11 DECLARE uname VARCHAR(30) DEFAULT '';
12 loop_label:LOOP
13 SET idx=idx+1;
14 select studentname into uname from students where studentid = idx;
15 /*如果遇到studentname为lala的同学,结束循环*/
16 IF uname='lala' THEN
17 LEAVE loop_label;
18 /*如果idx为偶数,则跳过本次循环*/
19 ELSEIF idx%2<>0 THEN
20 ITERATE loop_label;
21 /*这边加一个终结计数跳出的条件*/
22 ELSEIF idx>varial_count THEN
23 LEAVE loop_label;
24 END IF;
25 INSERT into studentCount values (idx,uname);
26
27 END LOOP;
28 END $
29 /*结束符置为;*/
30 DELIMITER ; 

调用存储过程,并输出你需要的数据:

 1 mysql> CALL sp_loop(6);
2 Query OK, 1 row affected
3
4 mysql> select * from studentCount;
5 +-----------+-------------+
6 | studentid | studentname |
7 +-----------+-------------+
8 | 2 | helen |
9 | 4 | sol |
10 | 6 | b2 |
11 +-----------+-------------+
12 3 rows in set

总结

1、了解了IF函数,它常用在SELECT语句中,类似于C#中的三元表达式。
2、IF条件表达式,类似于C#中的IF... ELSE...,多用于函数或存储过程中的判断选择逻辑。
3、了解CASE语句的两种用法,一种用在SELECT中使用,一种用在函数和存储过程中。
4、了解了三种循环体的使用,while、repeat分别对应C#中的while 和 do while循环,loop类似于一个while(true)的死循环。
5、循环体都包含在begin end中,循环体的控制依靠leave和iterate,leave相当于break,即退出整个循环体,iterate类似于continue,即跳过本次循环。

MySQL全面瓦解20:可编程性之流程控制语句的更多相关文章

  1. 关系数据库SQL之可编程性事务

    前言 前面关系数据库SQL之可编程性函数(用户自定义函数)一文提到关系型数据库提供了可编程性的函数.存储过程.事务.触发器及游标,前文已介绍了函数.存储过程,本文来介绍一下事务的使用.(还是以前面的银 ...

  2. 关系数据库SQL之可编程性存储过程

    前言 前面关系数据库SQL之可编程性函数(用户自定义函数)一文提到关系型数据库提供了可编程性的函数.存储过程.事务.触发器及游标,前文已介绍了函数,本文来介绍一下存储过程的创建.执行.删除.(还是以前 ...

  3. 关系数据库SQL之可编程性函数(用户自定义函数)

    前言 在关系型数据库中除了前面几篇基本的数据库和数据表操作之外,还提供了可编程性的函数.存储过程.事务.触发器及游标. 本文介绍的是函数. 函数分为两种: 系统函数 用户自定义函数 准备工作 这里以银 ...

  4. 《[MySQL技术内幕:SQL编程》读书笔记

    <[MySQL技术内幕:SQL编程>读书笔记 2019年3月31日23:12:11 严禁转载!!! <MySQL技术内幕:SQL编程>这本书是我比较喜欢的一位国内作者姜承尧, ...

  5. 关系数据库SQL之可编程性触发器

    前言 前面关系数据库SQL之可编程性函数(用户自定义函数)一文提到关系型数据库提供了可编程性的函数.存储过程.事务.触发器及游标,前文已介绍了函数.存储过程.事务,本文来介绍一下触发器的使用.(还是以 ...

  6. SQL Server 2008空间数据应用系列六:基于SQLCRL的空间数据可编程性

    原文:SQL Server 2008空间数据应用系列六:基于SQLCRL的空间数据可编程性 友情提示,您阅读本篇博文的先决条件如下: 1.本文示例基于Microsoft SQL Server 2008 ...

  7. mysql 5.6.20的安装、配置服务、设置编码格式

    一.安装 安装环境        系统:Window 32        版本:Mysql 5.6.20 1. 首先从官网上http://dev.mysql.com/downloads/mysql/ ...

  8. win 2012 安装mysql 5.7.20 及报错 This application requires Visual Studio 2013 Redistributable. Please install the Redistributable then run this installer again 的解决办法

    本文地址:http://www.cnblogs.com/jying/p/7764147.html    转载请注明出处. 安装过程其实挺简单,基本上下一步下一步,可以参考我的另一篇mysql安装文章: ...

  9. Windows 下 MySql 5.7.20安装及data和my.ini文件的配置(转)

    Windows 下 MySql 5.7.20安装及data和my.ini文件的配置     本文通过图文并茂的形式给大家介绍了MySql 5.7.20安装及data和my.ini文件的配置方法. my ...

  10. mysql 5.7.20 server status 是stopped的解决办法

    mysql 5.7.20 server status 是stopped的解决办法 在安装mysql 5.7.20的过程中,前几个过程都没什么问题,但是最后一个步骤就出问题了.当check一直提示con ...

随机推荐

  1. MySQL Fabric和MyBatis的整合过程中遇到的问题

    这是我昨天在整合MySQL Fabric和MyBatis时遇到的问题,花了大半天才解决的问题,解决的过程中在网上查找了很久,都没有找到解决的方案.现在记下来,希望能够帮助有同样问题的朋友.如果各位朋友 ...

  2. javascript鸭式辩型法实现接口

    <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/ ...

  3. 使用DocX开源组件,实现动态数据的填充。

    1.先解释一下,什么叫做动态数据,动态数据指的是,一条数据的格式固定,但数据的条数不固定. 2.应用环境,在一个表格当中如果,现在表格有三行n列,如果你需要在表格第一行后添加同等规格的一行或n行,应该 ...

  4. [BTS] Correct the specified Action, or refer to the documentation on the allowed formats for the Actions

    A message sent to adapter "WCF-SAP" on send port "CNILG.iHouse.SAP.WCFSAP" with ...

  5. cocos2dx 3.x以上版本搭建Mac环境(百分百可行)

    近期由于工作的原因,有机会接触了游戏行业,说实话,本人学程序最原始的初衷就是想做游戏,于是就创建了一篇cocos2d-x的分类来记录我在学习cocos2d-x的成长过程. 首先第一篇,想学cocos2 ...

  6. DataGridView 绑定 List

    DataGridView 绑定 List<T> 不会自动更新 正确方式是将  List<T> 设置为 BindingList<T> 即可 (双向绑定)

  7. 最经常使用的两种C++序列化方案的使用心得(protobuf和boost serialization)

    导读 1. 什么是序列化? 2. 为什么要序列化?优点在哪里? 3. C++对象序列化的四种方法 4. 最经常使用的两种序列化方案使用心得 正文 1. 什么是序列化? 程序猿在编写应用程序的时候往往须 ...

  8. Unity发布WebGL时如何修改默认的载入进度条

    Unity发布WebGL版本后,需要去除Unity的Logo,首先关闭Splash Image去除Made with Unity启动画面(在File->Build Settings->Pl ...

  9. 地图标绘系统V1.0测试版【申明:来源于网络】

    地图标绘系统V1.0测试版[申明:来源于网络] 地址:http://blog.csdn.net/allgis/article/details/39718085

  10. Unite Shanghai 2019全日程曝光(建议收藏)

    https://mp.weixin.qq.com/s/KvAyXpDhqWROtTX1Ol3a4Q 5月10-12日,Unite Shanghai 2019即将在上海国际会议中心正式开幕.本次大会共设 ...