一.MySql中Case-When的几种用法
注:本文章例子来自其他博客,整理后发出的。
1.语法
严格来讲不应该叫“mysql条件判断语句case when”他的专业术语应该叫“mysql
流程控制语句case语法”
用法1:
case变量
CASE case_value
WHEN when_value THEN statement_list
WHEN when_value2 THEN statement_list2
…
ELSE statement
END
用法2:
case表达式
CASE
WHEN search_condition THEN statement_list
WHEN search_condition2 THEN statement_list2
…
ELSE statement_list
END
语法说明:
对于第一种语法:
case_value 是一变量(字段名)。将该值与when_value每个when子句中的表达式进行比较,直到其中一个相等。当when_value找到平等时候,然后执行相应的then子statement_list。如果when_value都不相等则执行else子句statment_list
但是这个语法不能用于测试相等,因为null=null是错误的。
对于第二种语法:
将对每个when子句的search_condition表达式进行求值,直到其中一个为真,此时执行它对应then子句statement_list,如果search_conditson都不成立,则执行else
对于没有符合when_value 或search_condition的值,并且case语句不包含任何else
语句,测case语句会找不到case,那么这种情况如果是更新语句的话会将null
值更新进去
并且每个statement_list由一个或者多个sql组成,一个空的statment_list是不允许的
2.注意:
还有一个需要注意的问题,Case函数只返回第一个符合条件的值,剩下的Case部分将会被自动忽略。并且在有些情况下一定要写不符合when的情况 也就是else一定要写因为不符合的情况会自动填充为NULL值。
错误用法:
CASE col_1
WHEN 1 THEN ‘Right’
WHEN NULL THEN ‘Wrong’
END
在这个语句中When Null这一行总是返回unknown,所以永远不会出现Wrong的情况。因为这句可以替换成WHEN col_1 = NULL,这是一个错误的用法,这个时候我们应该选择用WHEN col_1 IS NULL。
3.例子
3.1例子(case变量式)
创建表:
CREATE TABLE county(
county varchar(20) NOT NULL,
population int(11) NOT NULL,
PRIMARY KEY(county)
)
添加数据:
INSERT INTO county VALUES('中国',600);
INSERT INTO county VALUES('美国',100);
INSERT INTO county VALUES('加拿大',100);
INSERT INTO county VALUES('英国',200);
INSERT INTO county VALUES('法国',300);
INSERT INTO county VALUES('日本',250);
INSERT INTO county VALUES('德国',200);
INSERT INTO county VALUES('墨西哥',50);
INSERT INTO county VALUES('印度',250);
有如下数据:(为了看得更清楚,我并没有使用国家代码,而是直接用国家名作为Primary Key)
国家(country) 人口(population)
中国 600
美国 100
加拿大 100
英国 200
法国 300
日本 250
德国 200
墨西哥 50
印度 250
根据这个国家人口数据,统计亚洲和北美洲的人口数量。应该得到下面这个结果。
洲 人口
亚洲 1100
北美洲 250
其他 700
如果使用Case函数,SQL代码如下:
SELECT SUM(population),
CASE country
WHEN '中国' THEN '亚洲'
WHEN '印度' THEN '亚洲'
WHEN '日本' THEN '亚洲'
WHEN '美国' THEN '北美洲'
WHEN '加拿大' THEN '北美洲'
WHEN '墨西哥' THEN '北美洲'
ELSE '其他' END
FROM Table_A
GROUP BY CASE country
WHEN '中国' THEN '亚洲'
WHEN '印度' THEN '亚洲'
WHEN '日本' THEN '亚洲'
WHEN '美国' THEN '北美洲'
WHEN '加拿大' THEN '北美洲'
WHEN '墨西哥' THEN '北美洲'
ELSE '其他' END;
结果:
3.2例子(case表达式式)
创建表:
CREATE TABLE country_sex (
id int(11) NOT NULL AUTO_INCREMENT,
country varchar(20) NOT NULL,
sex int(1) NOT NULL,
population int(11) NOT NULL,
primary key(id)
);
添加数据:
INSERT INTO country_sex VALUE(NULL,'中国',1,340);
INSERT INTO country_sex VALUES(NULL,'中国',2,260);
INSERT INTO country_sex VALUES(NULL,'美国',1,45);
INSERT INTO country_sex VALUES(NULL,'美国',2,55);
INSERT INTO country_sex VALUES(NULL,'加拿大',1,51);
INSERT INTO country_sex VALUES(NULL,'加拿大',2,49);
INSERT INTO country_sex VALUES(NULL,'英国',1,40);
INSERT INTO country_sex VALUES(NULL,'英国',2,60);
所有数据:
id country sex(1:男2:女) population
2 中国 1 340
3 中国 2 260
4 美国 1 45
5 美国 2 55
6 加拿大 1 51
7 加拿大 2 49
8 英国 1 40
9 英国 2 60
-- 需求 按照国家和性别进行分组
国家 男 女
中国 340 260
美国 45 55
加拿大 51 49
英国 40 60
SQL:
SELECT
country,
SUM(
CASE WHEN sex = 1 THEN population
ELSE 0
END
) AS '男',
SUM(
CASE WHEN sex = 2 THEN population
ELSE 0
END
) AS '女'
FROM
country_sex
GROUP BY
country;
结果:
3.3例子(嵌套式case用法)
假设有下面一个表
学号(std_id) 课程ID(class_id) 课程名(class_name) 主修flag(main_class_flg)
100 1 经济学 Y
100 2 历史学 N
200 2 历史学 N
200 3 考古学 Y
200 4 计算机 N
300 4 计算机 N
400 5 化学 N
500 6 数学 N
有的学生选择了同时修几门课程(100,200)也有的学生只选择了一门课程(300,400,500)。选修多门课程的学生,要选择一门课程作为主修,主修flag里面写入 Y。只选择一门课程的学生,主修flag为N(实际上要是写入Y的话,就没有下面的麻烦事了,为了举例子,还请多多包含)。
现在我们要按照下面两个条件对这个表进行查询
只选修一门课程的人,返回那门课程的ID
选修多门课程的人,返回所选的主课程ID
如果使用Case函数,我们只要一条SQL语句就可以解决问题,具体如下所示
SELECT std_id,
CASE WHEN COUNT(*) = 1 --只选择一门课程的学生的情况
THEN MAX(class_id)
ELSE MAX(CASE WHEN main_class_flg = 'Y'
THEN class_id
ELSE NULL END
)
END AS main_class
FROM Studentclass
GROUP BY std_id;
运行结果
STD_ID MAIN_class
------ ----------
100 1
200 3
300 4
400 5
500 6