一、建表和数据准备

create table t_window(

name varchar2(32),

orderdate varchar2(32),

cost varchar2(32)

)

select * from t_window

insert all into t_window values('jack','2015-01-01','10')

into t_window values('tony','2015-01-02','15')

into t_window values('jack','2015-02-03','23')

into t_window values('tony','2015-01-04','29')

into t_window values('jack','2015-01-05','46')

into t_window values('jack','2015-04-06','42')

into t_window values('tony','2015-01-07','50')

into t_window values('jack','2015-01-08','55')

into t_window values('mart','2015-04-08','62')

into t_window values('mart','2015-04-09','68')

into t_window values('neil','2015-05-10','12')

into t_window values('mart','2015-04-11','75')

into t_window values('neil','2015-06-12','80')

into t_window values('mart','2015-04-13','94')

select 1 from dual;

commit;

二、实例

聚合函数+over

max

因为加上了order by 所以默认从起始行到当前行

select name,orderdate,cost,max(cost) over(partition by substr(orderdate,1,7) order by cost)

from t_window

假如说我们想要查询在2015年4月份购买过的顾客及总人数,我们便可以使用窗口函数去去实现

select name,count(*) over ()

from t_window

where substr(orderdate,1,7) = '2015-04'

可见其实在2015年4月一共有5次购买记录,mart购买了4次,jack购买了1次.事实上,大多数情况下,我们是只看去重后的结果的.针对于这种情况,我们有两种实现方式

第一种:distinct

select distinct name,count(*) over ()

from t_window

where substr(orderdate,1,7) = '2015-04'

第二种:group by

select name,count(*) over ()

from t_window

where substr(orderdate,1,7) = '2015-04'

group by name

partition by子句

Over子句之后第一个提到的就是Partition By.Partition By子句也可以称为查询分区子句,非常类似于Group By,

都是将数据按照边界值分组,而Over之前的函数在每一个分组之内进行,如果超出了分组,则函数会重新计算.

实例

我们想要去看顾客的购买明细及月购买总额,可以执行如下的sql

select name,orderdate,cost,sum(cost) over(partition by substr(orderdate,1,7))

from t_window

order by子句

上述的场景,假如我们想要将cost按照月进行累加.这时我们引入order by子句.

order by子句会让输入的数据强制排序(文章前面提到过,窗口函数是SQL语句最后执行的函数,因此可以把SQL结果集想象成输入数据)。

Order By子句对于诸如Row_Number(),Lead(),LAG()等函数是必须的,因为如果数据无序,这些函数的结果就没有任何意义。因此如果有了Order By子句,

则Count(),Min()等计算出来的结果就没有任何意义。

(order by默认情况下聚合从起始行当当前行的数据)

select name,orderdate,cost,sum(cost) over(partition by substr(orderdate,1,7) order by orderdate )

from t_window

window子句

我们在上面已经通过使用partition by子句将数据进行了分组的处理.如果我们想要更细粒度的划分,我们就要引入window子句了

我们首先要理解两个概念:

如果只使用partition by子句,未指定order by的话,我们的聚合是分组内的聚合.

使用了order by子句,未使用window子句的情况下,默认从起点到当前行.

当同一个select查询中存在多个窗口函数时,他们相互之间是没有影响的.每个窗口函数应用自己的规则.

window子句:

PRECEDING:往前

FOLLOWING:往后

CURRENT ROW:当前行

UNBOUNDED:起点,UNBOUNDED PRECEDING 表示从前面的起点, UNBOUNDED FOLLOWING:表示到后面的终点

我们按照name进行分区,按照购物时间进行排序,做cost的累加.

如下我们结合使用window子句进行查询

select name,orderdate,cost,

sum(cost) over() as sample1,--所有行相加

sum(cost) over(partition by name) as sample2,--按name分组,组内数据相加

sum(cost) over(partition by name order by orderdate) as sample3,--按name分组,组内数据累加

sum(cost) over(partition by name order by orderdate rows between UNBOUNDED PRECEDING and current row ) as sample4 ,--和sample3一样,由起点到当前行的聚合

sum(cost) over(partition by name order by orderdate rows between 1 PRECEDING and current row) as sample5, --当前行和前面一行做聚合

sum(cost) over(partition by name order by orderdate rows between 1 PRECEDING AND 1 FOLLOWING ) as sample6,--当前行和前边一行及后面一行

sum(cost) over(partition by name order by orderdate rows between current row and UNBOUNDED FOLLOWING ) as sample7 --当前行及后面所有行

from t_window;

NTILE 窗口函数中的序列函数

NTILE(n),用于将分组数据按照顺序切分成n片,返回当前切片值

NTILE不支持ROWS BETWEEN,

比如 NTILE(2) OVER(PARTITION BY cookieid ORDER BY createtime ROWS BETWEEN 3 PRECEDING AND CURRENT ROW)

如果切片不均匀,默认增加第一个切片的分布

这个函数用什么应用场景呢?假如我们想要每位顾客购买金额前1/3的交易记录,我们便可以使用这个函数.

select name,orderdate,cost,

ntile(4) over(order by name) as sample1 , --全局数据切片

ntile(4) over(partition by name order by name), -- 按照name进行分组,在分组内将数据切成3份

ntile(4) over(order by cost),--全局按照cost升序排列,数据切成3份

ntile(4) over(partition by name order by cost ) --按照name分组,在分组内按照cost升序排列,数据切成3份

from t_window

row_number、rank、dense_rank 排名函数

这三个窗口函数的使用场景非常多

row_number()从1开始,按照顺序,生成分组内记录的序列,row_number()的值不会存在重复,当排序的值相同时,按照表中记录的顺序进行排列

RANK() 生成数据项在分组中的排名,排名相等会在名次中留下空位

DENSE_RANK() 生成数据项在分组中的排名,排名相等会在名次中不会留下空位

使用rank over()的时候,空值是最大的,如果排序字段为null, 可能造成null字段排在最前面,影响排序结果。

可以这样: rank over(partition by course order by score desc nulls last)

排名函数必须有包含 ORDER BY 的 OVER 子句,分组内从1开始排序。

select name,orderdate,cost,

row_number() over(partition by name order by substr(orderdate,1,7)) sample1,

rank() over(partition by name order by substr(orderdate,1,7)) sample2,

dense_rank() over(partition by name order by substr(orderdate,1,7)) sample3

from t_window

–分区cookieid pv相等则体现出函数的效果

建议partition字段的数据要比order by字段的数据条数多也最好不要同时使用在同一个字段

SELECT

cookieid,

createtime,

pv,

RANK() OVER(PARTITION BY cookieid ORDER BY pv desc) AS rn1,

DENSE_RANK() OVER(PARTITION BY cookieid ORDER BY pv desc) AS rn2,

ROW_NUMBER() OVER(PARTITION BY cookieid ORDER BY pv DESC) AS rn3

FROM lxw1234

WHERE cookieid = 'cookie1';

cookieid day pv rn1 rn2 rn3

cookie1 2015-04-12 7 1 1 1

cookie1 2015-04-11 5 2 2 2

cookie1 2015-04-15 4 3 3 3

cookie1 2015-04-16 4 3 3 4

cookie1 2015-04-13 3 5 4 5

cookie1 2015-04-14 2 6 5 6

cookie1 2015-04-10 1 7 6 7

LAG和LEAD函数

这两个函数为常用的窗口函数,可以返回上下数据行的数据.

以我们的订单表为例,假如我们想要查看顾客上次的购买时间可以这样去查询

select name,orderdate,cost,

lag(orderdate,1,'1900-01-01') over(partition by name order by orderdate ) as time1,

lag(orderdate,2) over (partition by name order by orderdate) as time2

from t_window;

time1取的为按照name进行分组,分组内升序排列,取上一行数据的值.

time2取的为按照name进行分组,分组内升序排列,取上面2行的数据的值,注意当lag函数为设置行数值时,默认为1行.未设定取不到时的默认值时,取null值.

lead函数与lag函数方向相反,取向下的数据

first_value取分组内排序后,截止到当前行,第一个值

last_value取分组内排序后,截止到当前行,最后一个值

order by截止到当前行

select name,orderdate,cost,

first_value(orderdate) over(partition by name order by orderdate) as time1,

last_value(orderdate) over(partition by name order by orderdate) as time2

from t_window

sum 开窗函数 oracle,oracle开窗函数案例详解相关推荐

  1. oracle rac 环境配置文件,学习笔记:Oracle RAC spfile参数文件配置案例详解

    天萃荷净 rac中的spfile探讨,记录一下Oracle RAC搭建完成后关于spfile参数文件的配置案例,与更改RAC环境中参数文件的方法 今天朋友的的rac,因为被同事做数据库升级,分别在两个 ...

  2. pandas:案例详解 rename函数 修改列名和行名

    pandas:案例详解rename函数 修改列名和索引 rename函数简介 0 构建学习数据 1 修改索引两种方式 2 修改列名两种方式 3 是否替换原列表 3 pandas 字母转换大小写 3 使 ...

  3. oracle数据库中索值,Oracle数据库中的索引详解

    Oracle数据库中的索引详解以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容,让我们赶快一起来看一下吧! 一 ROWID的概念 存储了row在数据文 ...

  4. oracle out参数查询,Oracle的out参数实例详解

    Oracle的out参数实例详解 一 概念 1.一般来讲,存储过程和存储函数的区别在于存储函数可以有一个返回值:而存储过程没有返回值. 2.过程和函数都可以通过out指定一个或多个输出行.我们可以利用 ...

  5. oracle的userenv和nls_lang详解

    oracle的userenv和nls_lang详解 1.userenv最常见的使用 userenv函数返回当前会话(session)的相关信息.以下sql语句可以查询当前会话连接的数据库字符集 sel ...

  6. oracle select执行顺序,oracle select执行顺序的详解

    oracle select执行顺序的详解 SQL Select语句完整的执行顺序:1.from子句组装来自不同数据源的数据: 2.where子句基于指定的条件对记录行进行筛选: 3.group by子 ...

  7. 删除oracle数据库的三种方法,oracle数据库的删除方法详解

    oracle数据库的删除方法详解 1.图形界面删除 练习之前记得创建快照 执行命令之前要保证数据库属于open状态 SQL> alter database open; [oracle@local ...

  8. ORACLE的索引和约束详解数据库

    ORACLE的索引和约束详解数据库 Oracle的约束 * 如果某个约束只作用于单独的字段,即可以在字段级定义约束,也可以在表级定义约束,但如果某个约束作用于多个字段, 必须在表级定义约束 * 在定义 ...

  9. Oracle之DBMS_SQL包用法详解

    Oracle之DBMS_SQL包用法详解 原文  http://zhangzhongjie.iteye.com/blog/1948093 通常运用 DBMS_SQL 包一般分为 如下 几步: 1. o ...

  10. oracle存储过程报错 跳过,oracle调试存储过程的过程详解

    oracle调试存储过程的过程详解 oracle如果存储过程比较复杂,我们要定位到错误就比较困难,那么可以存储过程的调试功能 先按简单的存储过程做个例子,就是上次做的存储过程(proc_test) 1 ...

最新文章

  1. DFS、栈、双向队列:CF264A- Escape from Stones
  2. windows下php配置redis
  3. html中源文件回车效果无效,网页制作使用教程第2节初级.ppt
  4. oracle 怎么调整emctl,emca和emctl的日常用法及实例(经验保存)
  5. Openwrt 串口密码登入【ZT】
  6. R语言分类算法之线性判别分析(Linear Discriminant Analysis)
  7. (7)Redis-Cluster集群理论及实践【上】
  8. java窗口设计_java简单窗口设计(源代码)
  9. 数字麦克风PDM信号采集与STM32 I2S接口应用(三)
  10. Android群英传笔记——摘要,概述,新的出发点,温故而知新,可以为师矣!
  11. 微信公众号如何上传html5,微信平台公众号怎么上传视频? 视频支持哪些格式
  12. Linux 应用程序安装卸载
  13. Robot Framework Selenium UI自动化测试 --- 实战篇
  14. Don‘t Stop Pretraining: Adapt Language Models to Domains and Tasks
  15. 瓦片地图面面观之投影
  16. 【大数据之Linux】
  17. 大数据培训课程数据清洗案例实操-简单解析版
  18. larbin的详细配置
  19. Dapp开发教程一 Asch Dapp Hello World
  20. 深度掌握 Java Stream 流操作,让你的代码高出一个逼格

热门文章

  1. 2013生活与学习齐头并进
  2. 联想B450笔记本拆解步骤
  3. SPI和QSPI协议学习
  4. SPI和IIC通信区别
  5. sysctl 默认值_/etc/sysctl.conf各个参数说明
  6. lol自动刷人机辅助_为何王者到了LOL手游连黄铜都未必能打过?其因有三
  7. Appium自动化测试1:Windows Appium环境安装
  8. idea:Unresolved plugin
  9. 微信小程序授权登录三种实现方式
  10. 财务数据分析sql python_我是如何从会计转行到数据分析