永发信息网

oracle中的over函数怎么用的,

答案:1  悬赏:0  手机版
解决时间 2021-05-24 08:26
oracle中的over函数怎么用的,
最佳答案

Oracle从8.1.6开始提供分析函数,分析函数用于计算基于组的某种聚合值,它和聚合函数的不同之处是对于每个组返回多行,而聚合函数对于每个组只返回一行.
一、 over函数
over函数指定了分析函数工作的数据窗口的大小,这个数据窗口大小可能会随着行的变化而变化,例如:
over(order by salary)按照salary排序进行累计,order by是个默认的开窗函数
over(partition by deptno) 按照部门分区
over(order by salary range between 50 preceding and 150 following)每行对应的数据窗口是之前行幅度值不超过50,之后行幅度值不超过150的数据记录
over(order by salary rows between 50 perceding and 150 following)前50行,后150行
over(order by salary rows between unbounded preceding and unbounded following)所有行
over(order by salary range between unbounded preceding and unbounded following)所有行
二、 sum函数
功能描述:该函数计算组中表达式的累积和.
SAMPLE:下例计算同一经理下员工的薪水累积值
SELECt manager_id, last_name, salary,
SUM (salary) OVER (PARTITION BY manager_id ORDER BY salary
RANGE UNBOUNDED PRECEDING) l_csum
FROM employees
WHERe manager_id in (101,103,108);
三、 应用实例
1, 测试环境设置
设有销售表t_sales (subcompany,branch,region,customer,sale_qty); 存储客户的销售明细,记录如下所示.
Subcompany Branch Region Customer Sale_qty
北京分公司 北京经营部 片区1 客户1 1
北京分公司 北京经营部 片区1 客户1 1
北京分公司 北京经营部 片区1 客户2 1
北京分公司 北京经营部 片区1 客户2 1
北京分公司 北京经营部 片区2 客户1 1
北京分公司 北京经营部 片区2 客户1 1
北京分公司 北京经营部 片区2 客户2 1
北京分公司 北京经营部 片区2 客户2 1
北京分公司 其他经营部 片区1 客户1 1
北京分公司 其他经营部 片区1 客户1 1
北京分公司 其他经营部 片区1 客户2 1
北京分公司 其他经营部 片区1 客户2 1
北京分公司 其他经营部 片区2 客户1 1
北京分公司 其他经营部 片区2 客户1 1
北京分公司 其他经营部 片区2 客户2 1
北京分公司 其他经营部 片区2 客户2 1
create table t_sales(
subcompany varchar2(40),
branch varchar2(40),
region varchar2(40),
customer varchar2(40),
sale_qty numeric(18,4)
);
comment on table t_sales is '销售表,分析函数测试';
comment on column t_sales.subcompany is '分公司';
comment on column t_sales.branch is '经营部';
comment on column t_sales.region is '片区';
comment on column t_sales.customer is '客户';
comment on column t_sales.sale_qty is '销售数量';
2,问题提出
现在要求给出销售汇总报表,报表中需要提供的数据包括客户汇总,和客户在其上级机构中的销售比例.
Subcompany Branch Region Customer Sale_qty Rate
北京分公司 北京经营部 片区1 客户1 2 50%
北京分公司 北京经营部 片区1 客户2 2 50%
北京分公司 北京经营部 片区1 小计 4 50%
北京分公司 北京经营部 片区2 客户1 2 50%
北京分公司 北京经营部 片区2 客户2 2 50%
北京分公司 北京经营部 片区2 小计 4 50%
北京分公司 北京经营部 小计 小计 8 50%
北京分公司 北京经营部 片区1 客户1 2 50%
北京分公司 北京经营部 片区1 客户2 2 50%
北京分公司 北京经营部 片区1 小计 4 50%
北京分公司 北京经营部 片区2 客户1 2 50%
北京分公司 北京经营部 片区2 客户2 2 50%
北京分公司 北京经营部 片区2 小计 4 50%
北京分公司 北京经营部 小计 小计 8 50%
北京分公司 小计 小计 小计 16 100%
3,解决方案(方案1)
首先我们可以使用oracle对group by 的扩展功能rollup得到如下的聚合汇总结果.
select
subcompany,
branch,
region,
customer,
sum(sale_qty) sale_qty
from t_sales
group by rollup(subcompany,branch,region,customer);
Subcompany Branch Region Customer Sale_qty
北京分公司 北京经营部 片区1 客户1 2
北京分公司 北京经营部 片区1 客户2 2
北京分公司 北京经营部 片区1 4
北京分公司 北京经营部 片区2 客户1 2
北京分公司 北京经营部 片区2 客户2 2
北京分公司 北京经营部 片区2 4
北京分公司 北京经营部 8
北京分公司 其他经营部 片区1 客户1 2
北京分公司 其他经营部 片区1 客户2 2
北京分公司 其他经营部 片区1 4
北京分公司 其他经营部 片区2 客户1 2
北京分公司 其他经营部 片区2 客户2 2
北京分公司 其他经营部 片区2 4
北京分公司 其他经营部 8
北京分公司 16
16
分析上面的临时结果,我们看到:
明细到客户的汇总信息,其除数为当前的sum(sale_qty),被除数应该是到片区的小计信息.
明细到片区的汇总信息,其除数为片区的sum(sale_qty),被除数为聚合到经营部的汇总数据.
.
考虑到上述因素,我们可以使用oracle的开窗函数over,将数据定位到我们需要定位的记录.如下代码中,我们利用开窗函数over直接将数据定位到其上次的小计位置.
over(partition by decode(f_branch, 1, null, subcompany), decode(f_branch, 1, null, decode(f_region, 1, null, branch)), decode(f_branch, 1, null, decode(f_region, 1, null, decode(f_customer, 1, null, region))), null)
经整理后的查询语句如下.
select subcompany,
decode(f_branch, 1,subcompany||'(С¼Æ)', branch),
decode(f_region,1,branch||'(С¼Æ)',region),
decode(f_customer,1,region||'(С¼Æ)', customer),
sale_qty,
trim(to_char(round(sale_qty/
sum(sale_qty) over(partition by decode(f_branch, 1, null, subcompany), decode(f_branch, 1, null, decode(f_region, 1, null, branch)), decode(f_branch, 1, null, decode(f_region, 1, null, decode(f_customer, 1, null, region))), null),2) *100,99990.99))
from (select grouping(branch) f_branch,
grouping(region) f_region,
grouping(customer) f_customer,
subcompany,
branch,
region,
customer,
sum(sale_qty) sale_qty
from t_sales
group by subcompany, rollup(branch, region, customer))
Subcompany Branch Region Customer Sale_qty Rate
北京分公司 北京经营部 片区1 客户1 2 50.00
北京分公司 北京经营部 片区1 客户2 2 50.00
北京分公司 北京经营部 片区2 客户1 2 50.00
北京分公司 北京经营部 片区2 客户2 2 50.00
北京分公司 北京经营部 片区1 片区1(小计) 4 50.00
北京分公司 北京经营部 片区2 片区2(小计) 4 50.00
北京分公司 其他经营部 片区1 客户1 2 50.00
北京分公司 其他经营部 片区1 客户2 2 50.00
北京分公司 其他经营部 片区2 客户1 2 50.00
北京分公司 其他经营部 片区2 客户2 2 50.00
北京分公司 其他经营部 片区1 片区1(小计) 4 50.00
北京分公司 其他经营部 片区2 片区2(小计) 4 50.00
北京分公司 北京经营部 北京经营部(小计) (小计) 8 50.00
北京分公司 其他经营部 其他经营部(小计) (小计) 8 50.00
北京分公司 北京分公司(小计) (小计) (小计) 16 100.00
北京分公司 北京经营部 片区1 客户1 2 50.00
4,可能的另外一种解决方式(方案2)
select subcompany,
decode(f_branch, 1,subcompany||'(С¼Æ)', branch),
decode(f_region,1,branch||'(С¼Æ)',region),
decode(f_customer,1,region||'(С¼Æ)', customer),
sale_qty,

decode(f_branch+f_region+f_customer,
0,
(sum(sale_qty) over(partition by subcompany,branch,region))/2,
1,
(sum(sale_qty) over(partition by subcompany,branch))/3,
2,
(sum(sale_qty) over(partition by subcompany))/4 ,
sum(sale_qty) over()/4
)
from (select grouping(branch) f_branch,
grouping(region) f_region,
grouping(customer) f_customer,
subcompany,
branch,
region,
customer,
sum(sale_qty) sale_qty
from t_sales
group by subcompany, rollup(branch, region, customer))
在上面的解决方式中,最大的问题在于开窗函数过大.导致每次计算涉及到的行数过多,影响到执行的速度和效率.并且需要额外的计算处理清除多余叠加进去的数值 .


我要举报
如以上问答信息为低俗、色情、不良、暴力、侵权、涉及违法等信息,可以点下面链接进行举报!
大家都在看
我想问一下 为什么和男朋友发生了关性月经终
我想学吉他·但一点也不了解·请问有资料给我
高一数学 ,急!
让胡路区国家电网大庆郊区农电局客户报修中心
http://streamrdt.music.qq.com/8529.79f0871
碾子山区齐齐哈尔瑞德兴旅店在什么地方啊,我
四川九华药品零售连锁有限公司新都普济药房地
要怎样QQ密码才不会被别人盗呢?
淇滨区鹤壁绿竹文化中心哪位知道具体地址啊
为什么我手机每天都收到好几条“我正在使用你
IP地址和物理地址的关系
监利县荆州米奇国际幼儿园这个地址怎么能查询
邵东县邵阳高岭小学怎么去啊,谁知道地址啊
甲车以3m/s²的加速度由静止做匀速直线运动,
35的旧式光影剑能卖钱啵?卖多少?
推荐资讯
芙蓉区楚留仙职业培训学校(长沙校区)这个地址
冰淇淋都有什么口味,美年达和粉达各是什么味
石鼓区衡阳Sweet D甜蜜独享在什么地方啊,我
东宝区荆门中国联通(家庭体验店)这个地址怎么
东方六加一把六个数全选中,位置不一样可以吗
湘阴县岳阳晚安家纺(湘阴县关公潭乡卫生院南)
CF会被封号吗?封成什么样子的?
阳光办公地址有知道的么?有点事想过去
怎样算幸运数字,一个人的幸运数字是怎么算的
移动手机怎么发短信充qb
用0到9这十个数字组成加,减,乘三道等式(每个
今天的神秘商品的链接是多少
正方形一边上任一点到这个正方形两条对角线的
阴历怎么看 ?