Oracle数据分摊问题解析
经常会碰到,由于业务需要,需要将某种汇总的数据按照一定的原则分摊给一堆数据。 其实,如果逻辑清晰的话,这类型的程序还是比较好些的。 本文重点是如果用简单的程序实现这种效果,而且不容易分摊分错。 所有的分摊问题,首先必须要搞清楚以下几点问题: 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 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 = 'P0000001'
AND MMT.transaction_date DBMS_OUTPUT.PUT_LINE('L_PRE_PERIOD_QTY:'||L_PRE_PERIOD_QTY||' -L_CURR_PERIOD_QTY:'||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) = 'Y'
AND MMT.ORGANIZATION_ID = 103
AND MMT.INVENTORY_ITEM_ID = 11783561
AND MTLN.LOT_NUMBER = 'P0000001'
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 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
||'-'|| L_shipment_consume_Tbl(I).PRIMARY_QUANTITY
||'-'|| L_shipment_consume_Tbl(I).consume_pre_qty
||'-'|| 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
*/

Heiße KI -Werkzeuge

Undresser.AI Undress
KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover
Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool
Ausziehbilder kostenlos

Clothoff.io
KI-Kleiderentferner

AI Hentai Generator
Erstellen Sie kostenlos Ai Hentai.

Heißer Artikel

Heiße Werkzeuge

Notepad++7.3.1
Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version
Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1
Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6
Visuelle Webentwicklungstools

SublimeText3 Mac-Version
Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Heiße Themen



Oracle -Umgebungsvariablen -Konfigurationshandbuch: Erstellen Sie eine Variable der Oracle_Home -Umgebungsvariable, wobei Sie auf das Oracle Home -Verzeichnis verweisen. Fügen Sie das Oracle -Binärverzeichnis der Pfadumgebungsvariable hinzu. Setzen Sie die Umgebungsvariable TNS_ADMIN (wenn die Datei mit TNS benannt ist). Überprüfen Sie die Umgebungsvariableneinstellungen, um sicherzustellen, dass die Ausgabe die festgelegten Variablen anzeigt.

Oracle bietet mehrere Deduplication -Abfragemethoden: Das bestimmte Schlüsselwort gibt für jede Spalte einen eindeutigen Wert zurück. Die Gruppe nach Klausel gruppiert die Ergebnisse und gibt für jede Gruppe einen nicht repetitiven Wert zurück. Das eindeutige Schlüsselwort wird verwendet, um einen Index mit nur eindeutigen Zeilen zu erstellen, und das Abfragen des Index wird automatisch dedupliziert. Die Funktion row_number () weist eindeutige Zahlen und filtert Ergebnisse aus, die nur Zeile 1 enthalten. Die Funktion min () oder max () gibt nicht repetitive Werte einer numerischen Spalte zurück. Der Schnittbetreiber gibt die gemeinsamen Werte der beiden Ergebnissätze zurück (keine Duplikate).

Oracle -verstümmelte Probleme können gelöst werden, indem der Datenbankzeichen gesetzt wird, um sicherzustellen, dass sie mit den Daten übereinstimmen. Legen Sie den Client -Zeichen für die Datenbank fest. Konvertieren Sie Daten oder ändern Sie Spaltenzeichen -Sets so, dass sie Datenbankzeichen -Sets übereinstimmen. Verwenden Sie Unicode -Zeichensätze und vermeiden Sie Multibyte -Zeichensätze. Überprüfen Sie, ob die Spracheinstellungen der Datenbank und des Clients korrekt sind.

Eine gespeicherte Prozedur ist eine Reihe von SQL -Anweisungen, die in einer Datenbank gespeichert werden können und wiederholt als separate Einheit bezeichnet werden können. Sie können Parameter (in, out, in Out) akzeptieren und die Vorteile von Code -Wiederverwendung, Sicherheit, Leistung und Modularität bereitstellen. Beispiel: Erstellen Sie eine gespeicherte Prozedur Calculate_sum, um die Summe von zwei Zahlen zu berechnen und sie im Out -Parameter zu speichern.

Um die Oracle -Tablespace -Größe abzufragen, führen Sie die folgenden Schritte aus: Bestimmen Sie den Namen Tablespace, indem Sie die Abfrage ausführen: Wählen Sie Tablespace_Name aus dba_tablespaces. Abfragen Sie die Tablespace -Größe durch Ausführen der Abfrage: Summe (Bytes) als Total_Size, sum (bytes_free) als verfügbare_space, sum

Verwenden Sie die Anweisung für die Änderung der Tabelle, die spezifische Syntax lautet wie folgt: Alter table table_name add column_name data_type [Constraint-Clause]. Wo: table_name der Tabellenname ist, Spalten_Name ist der Feldname, Data_Type ist der Datentyp, und die Einschränkungsklausel ist eine optionale Einschränkung. Beispiel: Änderung der Tabelle Mitarbeiter Fügen Sie der Mitarbeitertabelle eine E -Mail -Varchar2 (100) hinzu.

Ein Primärschlüssel ist eine spezielle Spalte oder Kombination von Spalten, die jede Zeile in einer Tabelle eindeutig identifizieren. Es stellt sicher, dass die Datensätze in der Tabelle eindeutig sind und erstellt werden können durch: Angabe des Tabellennamens mithilfe der Änderungstabelle Anweisung. Fügen Sie das Schlüsselwort der Primärschlüssel hinzu, gefolgt vom Spaltennamen, um als Primärschlüssel anzugeben. Primärschlüsseleinschränkungen tragen dazu bei, die Einzigartigkeit der Daten zu gewährleisten, die Abfragegeschwindigkeiten zu verbessern, doppelte Datensätze zu verhindern und die Tabellenverbindungen zu vereinfachen.

Die Schritte zur Änderung des Feldnamens in einer Oracle -Tabelle sind wie folgt: Verbindung zur Datenbank herstellen. Verwenden Sie die Anweisung für die Änderung der Tabelle, um den Feldnamen zu ändern. Geben Sie den Tabellennamen und den alten Feldnamen an. Geben Sie den neuen Feldnamen an. Änderungen einreichen.
