首页 > 代码库 > oracle--model用法

oracle--model用法

一. 什么是 model语句

model 语句是 Oracle10g 的新功能,
SQL中的MODEL从句是Oracle 10g数据库为企业智能领域引入的一个重要的新功能。这个从句对喜欢从Oracle中区出来,再放入电子表格进行分析的会计人员有很大帮助。SQL中的MODEL从句的目的就是让SQL语句拥有从普通的SELECT结果中创建多维数组的能力,然后再在这个SQL电子表格上进行行间或数组的计算。
MODEL从句通过讲一个查询中的各列映射为三组而定义一个多位数组,这三个组分别是分区(partitions)、维度(dimensions)和计量(measures)
对 SQL 的结果集进行处理。执行顺序是位于 Having 之后。

二. model 的使用场景

model 典型使用场景 。
1. 合计行追加
2. 行列变换
3. 使用当前行的前后行
4. RegExp_Replace 函数的循环执行

三. model语法

MODEL [RETURN [UPDATED | ALL] ROWS][reference models][PARTITION BY (<cols>)]DIMENSION BY (<cols>)MEASURES (<cols>) [IGNORE NAV] | [KEEP NAV][RULES[UPSERT | UPDATE][AUTOMATIC ORDER | SEQUENTIAL ORDER]

 例子说明:

CREATE TABLE A ASSELECT ‘lottu‘ AS vname,1 AS vals FROM dual;SELECT vname,vals FROM AMODEL--partition by ()可以忽略DIMENSION BY(vals) MEASURES(vname)    RULES (vname[1]=‘0924‘);

 输出结果:
0924    1

解释参数:

model:  model  语句的关键字,必须 。
partition by : 按照××分组
dimension by n dimension:  维度的意思,可以理解为数组的索引,必须 。
measures:  指定作为数组的列
rules: 对数组进行各种操作的描述

四. model return updated rows含义
说明:rules 的缺省行为是存在就更新,不存在则追加;使用 model return updated rows 的话,被 rules  更新或者插入的行才显示,没有更
新过的行不再作为 SQL  的结果

INSERT INTO A VALUES (‘LI‘,2);SELECT vname,vals FROM AMODEL RETURN UPDATED ROWSDIMENSION BY(vals)MEASURES(vname)RULES (vname[2]=‘0924‘);

 五. 举例说明:

例如:数据如下

CREATE TABLE B(p_id NUMBER,p_year Varchar2(5),p_val NUMBER);INSERT INTO B VALUES (1001,‘2011‘,25); INSERT INTO B VALUES (1001,‘2012‘,35); INSERT INTO B VALUES (1001,‘2013‘,65); INSERT INTO B VALUES (1001,‘2014‘,95); INSERT INTO B VALUES (1002,‘2011‘,25); INSERT INTO B VALUES (1002,‘2012‘,55); INSERT INTO B VALUES (1002,‘2013‘,75); INSERT INTO B VALUES (1002,‘2014‘,95); 

 1.预测2015的收入是前两年的总和。

SELECT * FROM B MODEL RETURN UPDATED ROWS PARTITION BY (p_id) DIMENSION BY (p_year) MEASURES (p_val) RULES (p_val[‘2015‘]=p_val[‘2014‘]+p_val[‘2013‘]);

2.预测2015年的营业收入数据。假设2015年1001公司营业收入是其前两年的总和,1002公司2015年的数据比其上年收入2倍。那么MODEL从句如下

SELECT * FROM B MODEL RETURN UPDATED ROWS DIMENSION BY (p_id,p_year) MEASURES (p_val) RULES (p_val[1001,‘2015‘]=p_val[1001,‘2013‘]+p_val[1001,‘2014‘],        P_val[1002,‘2015‘]=2 * p_val[1002,‘2014‘]);

 3.用BETWEEN和AND返回特定范围内的数据单元

 SELECT * FROM B MODEL RETURN UPDATED ROWS PARTITION BY (p_id) DIMENSION BY (p_year) MEASURES (p_val) RULES (p_val[‘2015‘]=sum(p_val)[p_year BETWEEN ‘2013‘ AND ‘2014‘]);

 4. 使用for in语句来返回特定范围内

 SELECT * FROM B MODEL RETURN UPDATED ROWS PARTITION BY (p_id) DIMENSION BY (p_year) MEASURES (p_val) RULES (p_val[‘2015‘]=sum(p_val)[for p_year in (‘2014‘,‘2013‘)]);
--如果 p_year 是数值类型,还可以用 for year from 2013 to 2014 increment 1 的语法,如果是其他类型,
还可以用在 in 子句带子查询的办法,比如 for p_year in (select year from B)

 5.用ANY和IS ANY访问所有的数据单元可以用ANY和IS ANY谓词访问数组中所有的数据单元。ANY和位置标记合用,IS ANY和符号标记合用
例如预测2017年的营业收入数据;是所有年份的总和;

SELECT * FROM B MODEL RETURN UPDATED ROWS PARTITION BY (p_id) DIMENSION BY (p_year) MEASURES (p_val) RULES (p_val[‘2017‘]=SUM(p_val)[ANY]);--或者any改为 p_year is any都可以。

 6.用CURRENTV()获取某个维度的当前值
  改写事例2

SELECT * FROM B MODEL RETURN UPDATED ROWS DIMENSION BY (p_id,p_year) MEASURES (p_val) RULES (p_val[1001,‘2015‘]=p_val[currentv(),‘2013‘]+p_val[currentv(),‘2014‘], P_val[1002,‘2015‘]=2 * p_val[currentv(),‘2014‘]);

 7.至于为什么要用到它;有什么好处?请看下面一例
 前不久我看到群里一个问题;找出p_color为‘red‘的p_value比为‘blue‘多的p_product
 这个题目不难;你用decode;或者case when,把blue的p_value判断为负;再利用组函数sum即可。
 若用model来写;你看有没有高大上的感觉。

select p_roduct,p_value (select * from tb  model return updated rows partition by (p_product) dimension by (p_color) measures (p_value) rules (p_value[‘red-bule‘]=p_value[‘red‘]-p_value[‘blue‘]))  from p_vlaue > 0;

 总结:功能很强大,语法太复杂了,

资料 --<pro oracle sql>第九章。

oracle--model用法