设为首页 收藏本站
查看: 1373|回复: 0

[经验分享] Oracle select case when

[复制链接]
YunVN网友  发表于 2018-9-6 13:24:09 |阅读模式
  Case具有两种格式。简单Case函数和Case搜索函数。
--简单Case函数CASE sexWHEN '1' THEN '男'WHEN '2' THEN '女'ELSE '其他' END--Case搜索函数CASE WHEN sex = '1' THEN '男'WHEN sex = '2' THEN '女'ELSE '其他' END  这两种方式,可以实现相同的功能。简单Case函数的写法相对比较简洁,但是和Case搜索函数相比,功能方面会有些限制,比如写判断式。
  还有一个需要注意的问题,Case函数只返回第一个符合条件的值,剩下的Case部分将会被自动忽略。
--比如说,下面这段SQL,你永远无法得到“第二类”这个结果CASE WHEN col_1 IN ( 'a', 'b') THEN '第一类'WHEN col_1 IN ('a')  THEN '第二类'ELSE'其他' END  下面我们来看一下,使用Case函数都能做些什么事情。
  一,已知数据按照另外一种方式进行分组,分析。
  有如下数据:(为了看得更清楚,我并没有使用国家代码,而是直接用国家名作为Primary Key)
国家(country)人口(population)中国600美国100加拿大100英国200法国300日本250德国200墨西哥50印度250  根据这个国家人口数据,统计亚洲和北美洲的人口数量。应该得到下面这个结果。
洲人口亚洲1100北美洲250其他700  想要解决这个问题,你会怎么做?生成一个带有洲Code的View,是一个解决方法,但是这样很难动态的改变统计的方式。
  如果使用Case函数,SQL代码如下:
SELECT  SUM(population),CASE countryWHEN '中国'     THEN '亚洲'WHEN '印度'     THEN '亚洲'WHEN '日本'     THEN '亚洲'WHEN '美国'     THEN '北美洲'WHEN '加拿大'  THEN '北美洲'WHEN '墨西哥'  THEN '北美洲'ELSE '其他' ENDFROM    Table_AGROUP BYCASE countryWHEN '中国'     THEN '亚洲'WHEN '印度'     THEN '亚洲'WHEN '日本'     THEN '亚洲'WHEN '美国'     THEN '北美洲'WHEN '加拿大'  THEN '北美洲'WHEN '墨西哥'  THEN '北美洲'ELSE '其他' END;  同样的,我们也可以用这个方法来判断工资的等级,并统计每一等级的人数。SQL代码如下;
SELECTCASE WHEN salary  500 AND salary  600 AND salary  800 AND salary  1000 )  女职员的条件倒是符合了,男职员就无法输入了。
  四,根据条件有选择的UPDATE。
  例,有如下更新条件

  •   工资5000以上的职员,工资减少10%
  •   工资在2000到4600之间的职员,工资增加15%
  很容易考虑的是选择执行两次UPDATE语句,如下所示
--条件1UPDATE Personnel  SET salary = salary * 0.9WHERE salary >= 5000;--条件2UPDATE Personnel  SET salary = salary * 1.15WHERE salary >= 2000 AND salary < 4600;  但是事情没有想象得那么简单,假设有个人工资5000块。首先,按照条件1,工资减少10%,变成工资4500。接下来运行第二个SQL时候,因 为这个人的工资是4500在2000到4600的范围之内, 需增加15%,最后这个人的工资结果是5175,不但没有减少,反而增加了。如果要是反过来执行,那么工资4600的人相反会变成减少工资。暂且不管这个 规章是多么荒诞,如果想要一个SQL 语句实现这个功能的话,我们需要用到Case函数。代码如下:
UPDATE PersonnelSET salary =CASE WHEN salary >= 5000                THEN salary * 0.9WHEN salary >= 2000 AND salary < 4600  THEN salary * 1.15ELSE salary END;  这里要注意一点,最后一行的ELSE salary是必需的,要是没有这行,不符合这两个条件的人的工资将会被写成NUll,那可就大事不妙了。在Case函数中Else部分的默认值是NULL,这点是需要注意的地方。
  这种方法还可以在很多地方使用,比如说变更主键这种累活。
  一般情况下,要想把两条数据的Primary key,a和b交换,需要经过临时存储,拷贝,读回数据的三个过程,要是使用Case函数的话,一切都变得简单多了。
p_keycol_1col_2a1张三b2李四c3王五  假设有如上数据,需要把主键a和b相互交换。用Case函数来实现的话,代码如下
UPDATE SomeTableSET p_key = CASE WHEN p_key = 'a'  THEN 'b'WHEN p_key = 'b'  THEN 'a'  ELSE p_key ENDWHERE p_key IN ('a', 'b');  同样的也可以交换两个Unique key。需要注意的是,如果有需要交换主键的情况发生,多半是当初对这个表的设计进行得不够到位,建议检查表的设计是否妥当。
  五,两个表数据是否一致的检查。
  Case函数不同于DECODE函数。在Case函数中,可以使用BETWEEN,LIKE,IS NULL,IN,EXISTS等等。比如说使用IN,EXISTS,可以进行子查询,从而 实现更多的功能。
  下 面具个例子来说明,有两个表,tbl_A,tbl_B,两个表中都有keyCol列。现在我们对两个表进行比较,tbl_A中的keyCol列的数据如果 在tbl_B的keyCol列的数据中可以找到, 返回结果'Matched',如果没有找到,返回结果'Unmatched'。
  要实现下面这个功能,可以使用下面两条语句
--使用IN的时候SELECT keyCol,CASE WHEN keyCol IN ( SELECT keyCol FROM tbl_B )  THEN 'Matched'ELSE 'Unmatched' END LabelFROM tbl_A;--使用EXISTS的时候SELECT keyCol,CASE WHEN EXISTS ( SELECT * FROM tbl_B  WHERE tbl_A.keyCol = tbl_B.keyCol )  THEN 'Matched'  ELSE 'Unmatched' END LabelFROM tbl_A;  使用IN和EXISTS的结果是相同的。也可以使用NOT IN和NOT EXISTS,但是这个时候要注意NULL的情况。
  六,在Case函数中使用合计函数
  假设有下面一个表
学号(std_id)课程ID(class_id)课程名(class_name)主修flag(main_class_flg)1001经济学Y1002历史学N2002历史学N2003考古学Y2004计算机N3004计算机N4005化学N5006数学N  有的学生选择了同时修几门课程(100,200)也有的学生只选择了一门课程(300,400,500)。选修多门课程的学生,要选择一门课程作 为主修,主修flag里面写入 Y。只选择一门课程的学生,主修flag为N(实际上要是写入Y的话,就没有下面的麻烦事了,为了举例子,还请多多包含)。
  现在我们要按照下面两个条件对这个表进行查询

  •   只选修一门课程的人,返回那门课程的ID
  •   选修多门课程的人,返回所选的主课程ID
  简单的想法就是,执行两条不同的SQL语句进行查询。
  条件1
--条件1:只选择了一门课程的学生  SELECT std_id, MAX(class_id) AS main_class  FROM Studentclass  GROUP BY std_id  HAVING COUNT(*) = 1;  执行结果1
STD_ID   MAIN_class300      4400      5500      6  条件2
--条件2:选择多门课程的学生  SELECT std_id, class_id AS main_class  FROM Studentclass  WHERE main_class_flg = 'Y' ;  执行结果2
STD_ID  MAIN_class100     1200     3  如果使用Case函数,我们只要一条SQL语句就可以解决问题,具体如下所示
SELECT  std_id,CASEWHEN COUNT(*) = 1  --只选择一门课程的学生的情况  THEN MAX(class_id)ELSEMAX(CASE WHEN main_class_flg = 'Y'  THEN class_id  ELSE NULL END  )END AS main_classFROM Studentclass  GROUP BY std_id;  运行结果
STD_ID   MAIN_class100      1200      3300      4400      5500      6  通过在Case函数中嵌套Case函数,在合计函数中使用Case函数等方法,我们可以轻松的解决这个问题。使用Case函数给我们带来了更大的自由度。
  最后提醒一下使用Case函数的新手注意不要犯下面的错误
CASE col_1WHEN  1        THEN 'Right'WHEN  NULL     THEN 'Wrong'END  在这个语句中When Null这一行总是返回unknown,所以永远不会出现Wrong的情况。因为这句实际表达的意思是
  WHEN col_1 = NULL,这是一个错误的用法,这个时候我们应该选择用WHEN col_1 IS NULL。
  ——————————————————————————————————————————————————————
  SELECT CASE WHEN 的用法
  select 与 case结合使用最大的好处有两点,一是在显示查询结果时可以灵活的组织格式,二是有效避免了多次对同一个表或几个表的访问。下 面举个简单的例子来说明。例如表 students(id, name ,birthday, sex, grade),要求按每个年级统计男生和女生的数量各是多少,统计结果的表头为,年级,男生数量,女生数量。如果不用select case when,为了将男女数量并列显示,统计起来非常麻烦,先确定年级信息,再根据年级取男生数和女生数,而且很容易出错。用select case when写法如下:
  SELECT   grade, COUNT (CASE WHEN sex = 1 THEN 1      /*sex 1为男生,2位女生*/
  ELSE NULL
  END) 男生数,
  COUNT (CASE WHEN sex = 2 THEN 1
  ELSE NULL
  END) 女生数
  FROM students GROUP BY grade;



运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.iyunv.com/thread-564119-1-1.html 上篇帖子: ORACLE 建库过程总结 下篇帖子: CentOS7.2安装oracle 12c RAC-ld0381的学习之旅
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表