热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Oracle数据分摊问题解析

经常会碰到,由于业务需要,需要将某种汇总的数据按照一定的原则分摊给一堆数据。其实,如果逻辑清晰的话,这类型的程序还是比较好些的。本文重点是如果用简单的程序实现这种效果,而且不容易分摊分错。所有的分摊问题,首先必须要搞清楚以下几点问题:1

经常会碰到,由于业务需要,需要将某种汇总的数据按照一定的原则分摊给一堆数据。 其实,如果逻辑清晰的话,这类型的程序还是比较好些的。 本文重点是如果用简单的程序实现这种效果,而且不容易分摊分错。 所有的分摊问题,首先必须要搞清楚以下几点问题: 1

经常会碰到,由于业务需要,需要将某种汇总的数据按照一定的原则分摊给一堆数据。
其实,如果逻辑清晰的话,这类型的程序还是比较好些的。
本文重点是如果用简单的程序实现这种效果,而且不容易分摊分错。

所有的分摊问题,首先必须要搞清楚以下几点问题:
1 首要的,要确定 什么东西,多少数量 分摊给什么东西?举个形象的例子,一桶沙子分摊给一些瓶子。
2 分摊的先后原则。上面的例子,一桶沙子分摊给一些瓶子,那瓶子的被分摊顺序是什么样子的?沙子先给哪个瓶子?要先确定清楚。

说得好像有点麻烦,举个例子说明。
最近接到的一个需求:
PO入库的时候,批次可能重复输入,所以入库之后,库存已经汇总在一起了。然后用户对(汇总的)库存进行消耗(就是杂发)。
现在需要有个报表可以知道:按照先进先出的原则,区分用户的一段期间内的消耗数量 对应的是那笔入库单号。

备注:假设下面的数量对应都是主单位。

7.1 入库单R1 料号A 批次P1 接收入库 400
7.3 入库单R2 料号A 批次P1 接收入库 300
这时候,P1批次库存共 700
-------消耗(杂发)明细
7.10 消耗P1 100
7.12 消耗P1 200
8.10 消耗P1 200
8.13 消耗P1 100
8.20 消耗P1 50
9.20 消耗P1 50


如果查询报表的日期选择的是:8.1~8.31
8.1号 之前共消耗100+200=300
8.1~8.31号 之内一共消耗:200+100+50=350

所以核心问题是要将350如何分摊在R1和R2里面。
要实现的分摊效果:
入库总数 之前消耗的分摊 期间内消耗的分摊
R1 400 300 100
R2 300 0 250

所以,结果是,报表是:8.1~8.31
一共消耗350,对应入库单的消耗情况:
R1消耗100
R2消耗250

实现逻辑:
你可以假想,现在有2个沙桶,
红色的沙桶装的沙子是 之前消耗的分摊 的数量
黑色的沙桶装的沙子是 期间内消耗的分摊 的数量
每张入库单就是一个瓶子,所以共有2个瓶子,R1和R2。现在是如何将 红色的沙子 和 黑色的沙子 装到这2个瓶子里面。

装沙规则:
1 用沙子的顺序:先用 红色的沙子,用完之后,再用黑色的沙子。
2 装瓶子的顺序:按照先进先出的原则,必须先装瓶子R1,再装R2.



DECLARE
L_PRE_PERIOD_QTY NUMBER; ---期间前的汇总消耗量 ---之前消耗的分摊 的数量---红色的沙子
L_CURR_PERIOD_QTY NUMBER ; --本期的汇总消耗量--期间内消耗的分摊 的数量---黑色的沙子
----装的结果用记录类型存下来,因为后面要用到。
TYPE shipment_consume_Rec_Type IS RECORD
(
SHIPMENT_LINE_ID NUMBER
, PRIMARY_QUANTITY NUMBER
, consume_pre_qty NUMBER
, consume_curr_qty NUMBER
);
TYPE shipment_consume_Tbl_Type IS TABLE OF shipment_consume_Rec_Type
INDEX BY BINARY_INTEGER ;
L_shipment_consume_Tbl shipment_consume_Tbl_Type;
N NUMBER;
BEGIN
----1 首先要算出红色的沙子和黑色的沙子的总数量,就是有多少数量可分摊。
SELECT nvl(sum(case when MMT.transaction_date <:P_F_START_DATE then
ABS(NVL(MTLN.PRIMARY_QUANTITY,0))
else
0
end ),0) PRE_PERIOD_QTY,
nvl(sum(case when MMT.transaction_date >= :P_F_START_DATE then
ABS(NVL(MTLN.PRIMARY_QUANTITY,0))
else
0
end ),0) CURR_PERIOD_QTY
INTO L_PRE_PERIOD_QTY,L_CURR_PERIOD_QTY
FROM MTL_MATERIAL_TRANSACTIONS MMT
,MTL_TRANSACTION_LOT_NUMBERS MTLN
WHERE MMT.TRANSACTION_ID = MTLN.TRANSACTION_ID
AND MMT.TRANSACTION_TYPE_ID = 74
AND MMT.TRANSACTION_ACTION_ID = 6
AND MMT.OWNING_TP_TYPE = 1 ---所有权转出的(寄售供应商的库存)
---
AND MMT.ORGANIZATION_ID = 103
AND MMT.INVENTORY_ITEM_ID = 11783561
AND MTLN.LOT_NUMBER = &#39;P0000001&#39;
AND MMT.transaction_date<=:P_F_END_DATE;
DBMS_OUTPUT.PUT_LINE(&#39;L_PRE_PERIOD_QTY:&#39;||L_PRE_PERIOD_QTY||&#39; -L_CURR_PERIOD_QTY:&#39;||L_CURR_PERIOD_QTY);
N := 1;

-----2 分摊主逻辑。
FOR REC_SHIPMENT_LINE IN (
-----瓶子(入库单)的游标
SELECT MMT.ORGANIZATION_ID
,MMT.INVENTORY_ITEM_ID
,MTLN.LOT_NUMBER
,MTLN.TRANSACTION_DATE
,RT.SHIPMENT_HEADER_ID
,RT.SHIPMENT_LINE_ID
,MTLN.PRIMARY_QUANTITY
FROM MTL_TRANSACTION_LOT_NUMBERS MTLN, MTL_MATERIAL_TRANSACTIONS MMT,RCV_TRANSACTIONS RT
WHERE MTLN.TRANSACTION_ID = MMT.TRANSACTION_ID
AND RT.TRANSACTION_ID = MMT.RCV_TRANSACTION_ID
AND MMT.TRANSACTION_TYPE_ID = 18
AND MMT.TRANSACTION_SOURCE_TYPE_ID = 1
AND XYG_PO_PKG.CHECK_PO_LINE_CONSIGN(RT.PO_LINE_ID) = &#39;Y&#39;
AND MMT.ORGANIZATION_ID = 103
AND MMT.INVENTORY_ITEM_ID = 11783561
AND MTLN.LOT_NUMBER = &#39;P0000001&#39;
ORDER BY MTLN.TRANSACTION_DATE,MMT.TRANSACTION_ID) LOOP
---2.1 优先消耗期初之前的耗料数量,就是先用红色的沙子的数量。
IF L_PRE_PERIOD_QTY >= REC_SHIPMENT_LINE.PRIMARY_QUANTITY THEN ---当红色沙子的数量大于瓶子的容量的时候。
L_shipment_consume_Tbl(N).SHIPMENT_LINE_ID :=REC_SHIPMENT_LINE.SHIPMENT_LINE_ID;
L_shipment_consume_Tbl(N).PRIMARY_QUANTITY :=REC_SHIPMENT_LINE.PRIMARY_QUANTITY;
----消耗红沙的数量就是瓶子的容量。
L_shipment_consume_Tbl(N).consume_pre_qty :=REC_SHIPMENT_LINE.PRIMARY_QUANTITY;
L_shipment_consume_Tbl(N).consume_curr_qty :=0;
----期初数量就是剩下要分配的数量。因为红色沙子已经被消耗掉一部分了。
L_PRE_PERIOD_QTY :=L_PRE_PERIOD_QTY-REC_SHIPMENT_LINE.PRIMARY_QUANTITY;
ELSE ---当红色沙子的数量小于瓶子容量的时候
L_shipment_consume_Tbl(N).SHIPMENT_LINE_ID :=REC_SHIPMENT_LINE.SHIPMENT_LINE_ID;
L_shipment_consume_Tbl(N).PRIMARY_QUANTITY :=REC_SHIPMENT_LINE.PRIMARY_QUANTITY;
----瓶子装 红色沙子的数量就是红色沙子的数量了
L_shipment_consume_Tbl(N).consume_pre_qty :=L_PRE_PERIOD_QTY;

----这时候已经用完红色沙子了,开始用黑色沙子了-----
---2.2 当黑色沙子数量大于瓶子 可用的容量 的时候。
IF L_CURR_PERIOD_QTY > (REC_SHIPMENT_LINE.PRIMARY_QUANTITY - L_PRE_PERIOD_QTY) THEN
---该瓶子 装黑色沙子的数量 就是 瓶子的可用容量。
L_shipment_consume_Tbl(N).consume_curr_qty := REC_SHIPMENT_LINE.PRIMARY_QUANTITY - L_PRE_PERIOD_QTY;
---本次还有多少数量需要被下一个单号分摊,就是确定剩下还有多少黑色沙子可用。
L_CURR_PERIOD_QTY := L_CURR_PERIOD_QTY - L_shipment_consume_Tbl(N).consume_curr_qty;
ELSE
----当黑色沙子数量 小于或者等于 瓶子的可用容量的时候
-----该瓶子装黑色沙子的数量就是 瓶子的可用容量。
L_shipment_consume_Tbl(N).consume_curr_qty := L_CURR_PERIOD_QTY;
-----黑色沙子用完咯!~~一定要赋值0,因为根据黑色沙子的使用情况判断后面是否要退出瓶子的循环。
L_CURR_PERIOD_QTY := 0;
END IF;
L_PRE_PERIOD_QTY:= 0;
END IF;
N := N+1;
----当黑色沙子用完的时候,要退出循环。因为沙子数量可能很少,但是瓶子很多。。。没必要再循环下去了。
IF L_CURR_PERIOD_QTY <= 0 THEN
EXIT;
END IF;
END LOOP;

---显示装的结果。
FOR I IN 1..L_shipment_consume_Tbl.COUNT LOOP
DBMS_OUTPUT.PUT_LINE(L_shipment_consume_Tbl(I).SHIPMENT_LINE_ID
||&#39;-&#39;|| L_shipment_consume_Tbl(I).PRIMARY_QUANTITY
||&#39;-&#39;|| L_shipment_consume_Tbl(I).consume_pre_qty
||&#39;-&#39;|| L_shipment_consume_Tbl(I).consume_curr_qty
);

END LOOP;
END;

/*
---例如:
L_PRE_PERIOD_QTY:0 -L_CURR_PERIOD_QTY:2020.2
18467366-1605.5-0-1605.5
18633076-5014.7-0-414.7

*/
推荐阅读
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • GetWindowLong函数
    今天在看一个代码里头写了GetWindowLong(hwnd,0),我当时就有点费解,靠,上网搜索函数原型说明,死活找不到第 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • PHP设置MySQL字符集的方法及使用mysqli_set_charset函数
    本文介绍了PHP设置MySQL字符集的方法,详细介绍了使用mysqli_set_charset函数来规定与数据库服务器进行数据传送时要使用的字符集。通过示例代码演示了如何设置默认客户端字符集。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 橱窗设计的表现手法及其应用
    本文介绍了橱窗设计的表现手法,包括直接展示、寓意与联想、夸张与幽默等。通过对商品的折、拉、叠、挂、堆等陈列技巧,橱窗设计能够充分展现商品的形态、质地、色彩、样式等特性。同时,寓意与联想可以通过象形形式或抽象几何道具来唤起消费者的联想与共鸣,创造出强烈的时代气息和视觉空间。合理的夸张和贴切的幽默能够明显夸大商品的美的因素,给人以新颖奇特的心理感受,引起人们的笑声和思考。通过这些表现手法,橱窗设计能够有效地传达商品的个性内涵,吸引消费者的注意力。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • faceu激萌变老特效的使用方法详解
    本文介绍了faceu激萌变老特效的使用方法,包括打开faceu激萌app、点击贴纸、选择热门贴纸中的变老特效,然后对准人脸进行拍摄,即可给照片添加变老特效。操作简单,适合新用户使用。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 大连微软技术社区举办《.net core始于足下》活动,获得微软赛百味和易迪斯的赞助
    九月十五日,大连微软技术社区举办了《.net core始于足下》活动,共有51人报名参加,实际到场人数为43人,还有一位专程从北京赶来的同学。活动得到了微软赛百味和易迪斯的赞助,场地也由易迪斯提供。活动中大家积极交流,取得了非常成功的效果。 ... [详细]
  • 给定一个二叉树,要求随机选择树上的一个节点。解法:遍历树的过程中,随机选择一个节点即可。具体做法参看:从输入 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文介绍了在微店中如何修改分销产品的价格以及设置价格的方法。客户在拍下商品后,在1小时内可以进行修改价格的操作,通过进入订单管理,点击未付款子项,可以找到订单信息并进行改价操作。修改价格后,买家会收到改价后的短信通知,在微店订单中进行付款即可。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
author-avatar
悠闲的大黄
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有