Oracle行转列 参数动态传入iBatis使用示例

最近做了一个需求,需要获取工作流数据的各个节点的渠道数量信息,各渠道的费用信息~

之前的需求是只需要获取渠道数据,所以做了渠道兼容,每个渠道数量的获取都是先case when 处理,然后再sum统计的

方案一:手动汇总数据为列数据(先case when 计算再sum统计)

例如:   #统计渠道数据量:渠道代码相同时,渠道数据计数1,不同时计数0

select
taskname
, case when t.CHANNEL_CODE = #channelcode0# then 1 else 0 end channel0
, case when t.CHANNEL_CODE = #channelcode1# then 1 else 0 end channel1
from tablename ;

  

完整的统计sql

select
tb.taskname
,sum(tb.channel0) channel0num
,sum(tb.channel1) channel1num
from (
  select
    taskname
    , case when t.CHANNEL_CODE = #channelcode0# then 1 else 0 end channel0
    , case when t.CHANNEL_CODE = #channelcode1# then 1 else 0 end channel1
  from tablename
)tb group by taskname 

执行效果图如下:

方案二:使用oracle的列转行函数

接下来我们看看强大的Oracle如何教我们行列转置,免得那么麻烦的去case when 再 sum了

 关键函数 pivot  (列转为行) 

在表名后面接以下这段 pivot( sum(colum_name) alisname for key_column  in (value1 name1, value2 name2))

 colum_name #需要统计的数据信息(渠道数据或者保费信息),这里需要用聚合函数

 alisname  #这个数据项的别名

 key_column #关键列,就是将此列数据转换为行

 value1:  key_column的可能值,这里如果是固定值可以直接写,在代码里面也可以用变量代替

 name1:  value1对应的别名,最后生成的数据名称会自动拼接为  name1_alisname  (例如如下的 ch0_cnt )

使用示例:

select
TASK_NAME
,sum(ch0_cnt) as channel0num
,sum(ch1_cnt) as channel1num
from tablename t pivot(count(CHANNEL_CODE) cnt for CHANNEL_CODE in (1 ch0, 21 ch1))
group by TASK_NAME

  

执行效果:(与方案一 殊途同归,但是这个会简洁很多,尤其是当渠道信息多的时候就会简洁的更明显了啦~)

行转列的时候统计多项数据:(可以写多个聚合函数,但是统计的中心列只有一项哦 以下示例均以 CHANNEL_CODE 渠道为中心,来统计各项数据)

select
TASK_NAME
,sum(ch0_cnt) as channel0num
,sum(ch1_cnt) as channel1num
,sum(ch0_daycount) as ch0_daycount
,sum(ch1_daycount) as ch1_daycount
from ReportNBUWData t pivot( sum(PASSED_TIME) daycount, count(CHANNEL_CODE) cnt for CHANNEL_CODE in (1 ch0, 21 ch1))
group by TASK_NAME

  

 方案三:ibatis使用动态变量

变量格式:<![CDATA[$channelcode0$]]>  ,其中 channelcode0 为变量名

千万要记得获取的变量的时候用 $parameter$ ,且要使用 <![CDATA[ ]]> 文本化

例如:

select
TASK_NAME
,sum(ch0_cnt) as channel0num
,sum(ch1_cnt) as channel1num
,sum(ch0_daycount) as ch0_daycount
,sum(ch1_daycount) as ch1_daycount
from ReportNBUWData t pivot( sum(PASSED_TIME) daycount, count(CHANNEL_CODE) cnt for CHANNEL_CODE in (<![CDATA[$channelcode0$]]> ch0,<![CDATA[$channelcode1$]]> ch1))
group by TASK_NAME

  

如果有多个变量参数,可以按照如下格式继续添加 : <![CDATA[$channelcode0$]]> ch0

需要注意的点:

1.入参不可以为空!!! 入参不可以为空!!!  一定要有具体的值!!!  意思就是 $channelcode0$ 变量对应的值不可以为空,也不可以为null!!!

2.要使用 <![CDATA[ ]]> 文本化  (否则就报错:pivot内不能使用动态变量)

3.不可以使用预编译 #channelcode0#,要使用 $channelcode0$!!! (否则就报错:pivot内不能使用动态变量)

参考博客:https://www.bbsmax.com/A/WpdK4oZnzV/ (oracle行转列、列转行、连续日期数字实现方式及mybatis下实现方式)

Oracle列转行 参数动态传入iBatis使用示例的更多相关文章

  1. Oracle列转行函数Listagg以及pivot查询示例

    简单的Oracle列转行函数Listagg示例: CREATE TABLE tbl_test (catalog VARCHAR(1),product VARCHAR(2),amount NUMBER) ...

  2. Oracle列转行函数版本不兼容解决方案

    业务场景 本博客记录一下Oracle列转行函数在Oracle11的一些不兼容问题,vm_concat在一些业务场景是必须的.不过这个函数使用要谨慎,底层实现应该也是group by等等实现的,性能并不 ...

  3. Oracle列转行函数LISTAGG()

    --Oracle列转行函数LISTAGG() with tb_temp as( select 'China' 国家,'Wuhan' 城市 from dual union all select 'Chi ...

  4. Oracle列转行函数使用

    一.业务场景 今天需要实现一个table,有一列的效果是:用户姓名A(账号a),用户姓名B(账号b)...这种格式.这就想到oracle的列转行函数vm_concat. 可以用类似这种格式wm_con ...

  5. oracle列转行

    unpivot()函数需要Oracle版本大于等于11g --创建表 create table Fruit(id int,name varchar(20), Q1 int, Q2 int, Q3 in ...

  6. Oracle 列转行函数 Listagg()

    这是最基础的用法: LISTAGG(XXX,XXX) WITHIN GROUP( ORDER BY XXX) 例: select listagg(oeid,',') within GROUP (ord ...

  7. oracle 列转行

    with temp as( as S3 from dual union all as S3 from dual ) select * from temp unpivot(Qty for Sizes i ...

  8. oracle列转行 WM_CONCAT LISTAGG

    开发给个SQL说给某个条件时报ORA-22922 代码段: SELECT 袋号, SUM(实际重量) AS 实际重量, SUM(材积重量) AS 材积重量, COUNT(运单号) AS 件数, TO_ ...

  9. oracle 行转列、列转行

    最近做数据处理,经常遇到需要行转列.列转行的场景,记录个非常简单实用的oracle  列转行.行转的列方法 1.行转列,基础数据如下 做行转列处理 处理SQL select user_name,max ...

随机推荐

  1. 团队项目-Beta冲刺(第一周)

    团队项目-Beta冲刺(第一周) 一. 作业描述 这个作业属于哪个课程 这个作业要求在哪里 团队名称 CTRL-IKun 这个作业的目标 合理安排时间完成接下来的任务,剩余任务预估,分配任务(开发,测 ...

  2. Spring框架学习笔记(8)——spring boot+mybatis plus+mysql项目环境搭建

    之前写的那篇Spring框架学习笔记(5)--Spring Boot创建与使用,发现有多小细节没有提及,,正好现在又学习了mybatis plus这款框架,打算重新整理一遍,并将细节说清楚 1.通过I ...

  3. epel-release的卸载重装

    1.yum remove epel-release 2.清空epel目录:rm -rf /var/cache/yum/x86_64/6/epel/ 3.安装,yum install  epel-rel ...

  4. Android教程2020 - RecyclerView获取滑动距离

    获取RecyclerView滑动的距离. Android教程2020 - 系列总览 本文链接 前面我们已经用RecyclerView显示一些数据. 本文演示如何获取RecyclerView的滑动距离. ...

  5. 在Anaconda3下安装(CPU版)TensorFlow(清华镜像源)

    1.打开Anaconda Prompt 2.搭建TensorFlow的环境: conda config --add channels https://mirrors.tuna.tsinghua.edu ...

  6. 【Qt学习笔记】Qt+VS2010的配置

    http://blog.csdn.net/jocyln9026/article/details/8575218 关于Qt Qt是1991年由Trolltech公司开发的一个跨平台的C++图形用户界面应 ...

  7. 对char *f=new char[4]赋初值

    用new 开辟的可以直接指向字符串常量,但是之后就不能进行修改了比如 char * f=new char[4]; f="ab"; f[0]='0'//错误 但是可以输出f[0]; ...

  8. css 浏览兼容问题及解决办法 (1)

    主流浏览器css兼容问题的总结 最近又搞了一波网站的兼容,由于要求ie浏览器还是要兼容到ie8,所以调起来还是各种蛋疼. 现在就post一些做兼容的总结,可能不够全面,但是可以告诉大家如何避过一些坑. ...

  9. 【译文连载】 理解Istio服务网格(第三章 流控)

    第3章 流控.............................................................................................. ...

  10. 《C# GDI+ 破境之道》:第一境 GDI+基础 —— 第三节:画圆形

    有了上一节画矩形的基础,画圆形就不要太轻松+EZ:)所以,本节在画边线及填充上,就不做过多的讲解了,关注一下画“随机椭圆”.“正圆”.“路径填充”的具体实现就好.与画矩形相比较,画椭圆与之完全一致,没 ...