说起 DB2 在线分析处理,可以用很好很强大来形容。这项功能特别适用于各种统计查询,这些查询用通常的SQL很难实现,或者根本就无发实现。首先,我们从一个简单的例子开始,来一步一步揭开它神秘的面纱,请看下面的SQL:
看到上面的ROW_NUMBER() OVER()了吗?很多人非常不理解,怎么两个函数能这么写呢?甚至有人怀疑上面的SQL语句是不是真的能执行。其实,ROW_NUMBER是个函数没错,它的作用从它的名字也可以看出来,就是给查询结果集编号。但是,OVER并不是一个函数,而是一个表达式,它的作用是定义一个作用域(或者可以说是结果集),OVER前面的函数只对OVER定义的结果集起作用。怎么样,不明白?没关系,我们后面还会详细介绍。
从上面的SQL我们可以看出,典型的 DB2 在线分析处理的格式包括两部分:函数部分和OVER表达式部分。那么,函数部分可以有哪些函数呢?如下:
上面这些函数的作用,我会在后面逐步给大家介绍,大家可以根据函数名猜测一下函数的作用。
假设我想在不改变上面语句的查询结果的情况下,追加对部门员工的平均工资和全体员工的平均工资的查询,怎么办呢?用通常的SQL很难查询,但是用OLAP函数则非常简单,如下SQL所示:
请注意序号和部门序号之间的区别,我们在查询部门序号的时候,在OVER表达式中多了两个子句,分别是PARTITION BY 和ORDER BY。它们有什么作用呢?在介绍它们的作用之前,我们先来回顾一下OVER的作用,还记得吗?
OVER是一个表达式,它的作用是定义一个作用域(或者可以说是结果集),OVER前面的函数只对OVER定义的结果集起作用。
ORDER BY的作用大家应该非常熟悉,用来对结果集排序。PARTITION BY的作用其实也很简单,和GROUP BY 的作用相同,用来对结果集分组。
到此为止,大家应该对OLAP函数的套路有一定的了解和体会了吧。大家看一下上面SQL的结果集,发现王五的工资是null,当我们按工资排序时,null被放到最后,我们想把null放在前边该怎么办呢?使用NULLS FIRST关键字即可,默认是NULLS LAST,请看下面的SQL:
请注意ROW_NUMBER和RANK之间的区别,RANK是等级,排名的意思,李四和李白的工资都是5000,他们并列排名第二。张三和王蓝的工资都是4000,怎么RANK函数的排名是第四,而DENSE_RANK的排名是第三呢?这正是这两个函数之间的区别。由于有两个第二名,所以RANK函数默认没有第三名。
现在又有个新问题,假设让你查询一下每个员工的工资以及工资小于他的所有员工的平均工资,该怎么办呢?怎么?没听明白问题?不要紧,请看下面的SQL:
上面SQL 中的OVER部分出现了一个ROWS子句,我们先来看一下ROWS子句的结构:
注意,以上关键字都是相对当前行的,UNBOUNDED PRECEDING表示当前行前面的所有行,也就是说没有上限;<number> PRECEDING表示从当前行开始到它前面的<number>行为止,例如,number=2,表示的是当前行前面的2行;CURRENT ROW表示当前行。至于其它两个关键字,我想,不用我说,你也应该知道了吧。如果你还不明白,请仔细分析上面SQL的查询结果。
OVER表达式还可以有个子句,那就是RANGE,它的使用方式和ROWS 十分相似,或者说一模一样,作用也差多不,不过有点区别,如下所示:
RANGE BETWEEN <上限条件> AND <下限条件>
其中的<上限条件> 、<下限条件>和ROWS一模一样,如下的SQL演示它们之间的区别:
上面SQL的RANGE 子句的作用是定义一个工资范围,这个范围的上限是当前行的工资-500,下限是当前行工资+500。例如:李四的工资是3000,所以上限是3000-500=2500,下限是3000+500=3500,那么有谁的工资在2500-3500这个范围呢?只有李四和李白,所以RANGE列的值就是3000(李四)+3200(李白)=6200。
小编还为您整理了以下内容,可能对您也有帮助:
DB2 分组后提取前5条数据
前5次,应该是按时间排序,不区分打卡地点,对吧?!
使用OLAP函数就能解决问题啦:
select * from (
select
(row_number() over (partition by 用户ID order by 打卡时间 asc)) AS row_id
用户ID,
打卡地点ID,
打卡时间
from table_name) t
where row_id<=5
DB2 分组后提取前5条数据
前5次,应该是按时间排序,不区分打卡地点,对吧?!
使用OLAP函数就能解决问题啦:
select * from (
select
(row_number() over (partition by 用户ID order by 打卡时间 asc)) AS row_id
用户ID,
打卡地点ID,
打卡时间
from table_name) t
where row_id<=5
DB2分页查询(DB2的分页查询到底该怎么写)
提起分页查询,除了那些还不知道什么是分页的人,大多数人的都会想到一个词,那就是 LIMIT ,不过很可惜, DB2 不支持这个关键字,那么 DB2 的分页查询到底该怎么写呢?只要你学会了 OLAP 函数,分页查询是非常简单的。即使你不会 OLAP 函数,按照下面的 SQL提起分页查询,除了那些还不知道什么是分页的人,大多数人的都会想到一个词,那就是LIMIT,不过很可惜,DB2不支持这个关键字,那么DB2的分页查询到底该怎么写呢?只要你学会了OLAP 函数,分页查询是非常简单的。即使你不会OLAP 函数,按照下面的SQL 照猫画虎也可以,如下:
DB2 分页
.CodeEntity .code_pieces ul.piece_anchor{width:25px;position:absolute;top:25px;left:-30px;z-index:1000;}
.CodeEntity .code_pieces ul.piece_anchor li{width:25px;background: #efe;margin-bottom:2px;}
.CodeEntity .code_pieces ul.piece_anchor li{border-left:3px #40AA63 solid;border-right:3px #efe solid;}
.CodeEntity .code_pieces ul.piece_anchor li:hover{border-right:3px #40AA63 solid;border-left:3px #efe solid;}
.CodeEntity .code_pieces ul.piece_anchor li a{color: #333;padding: 3px 10px;}
.CodeEntity .code_pieces .jump_to_code{visibility:hidden;position:relative;}
.CodeEntity .code_pieces .code_piece:hover .jump_to_code{visibility:visible;}
.CodeEntity .code_pieces .code_piece:hover .jump_to_code a{text-decoration:none;}
.CodeEntity .code_pieces h2 i{float:right;font-style:normal;font-weight:normal;}
.CodeEntity .code_pieces h2 i a{font-size:9pt;background: #FFFFFF;color:#00A;padding: 2px 5px;text-decoration:none;}
$velocityCount-
SELECT * FROM
(
SELECT B.*, ROWNUMBER() OVER() AS RN FROM
(
SELECT * FROM
) AS B
)AS A WHERE A.RN BETWEENAND ;