使用CTE递归将可拆分项目堆叠到垃圾箱中

时间:2019-07-09 09:09:26

标签: recursion cursor common-table-expression

我创建了一个简约的样本数据,目的是研究和展示这个有趣的任务。波纹管是用于设置测试表的代码。任务是将项目分发到预设箱中,同时:

  • 我们不能超过垃圾箱的最大容量
  • 如果(超出)剩余的存储箱容量,则可以(应该)对其进行拆分
  • 有一个垃圾箱(永远是第一个),它具有“无限”(非常高)的容量,任何不适合常规垃圾箱的东西都必须在这里结束。虽然此示例特有,但从广义上讲,它演示了如何处理CTE中的条件。
  • 我们开始填充从DueBin开始并向后向后(以降序排列)的项目。
  • 如果没有足够的物品可堆放,则
  • 箱可能会用尽全部容量,或者只能部分使用

Upper chart shows stacked items, bottom shows matching pre-set bins 上图显示堆积的物品,下图显示匹配的预设垃圾箱。使用CURSORS解决方案获取的结果

Bellow是使用CURSORS的完整的工作解决方案,但这是有关使用 CTE recursion 来完成此任务的问题。显然,我正在学习CTE并寻求性能提升,这在我以前的CTE递归应用中非常重要。

这是我创建的错误的CTE代码,我想它对您没有多大帮助。它远没有包括所有必需的逻辑,例如拆分项目或转移到具有可用容量的前一个容器。
我承认对CTE递归的经验很少,并且很难在此任务中使用它。

DECLARE @MinDate Date
SELECT @MinDate = MIN(BinDateTime) FROM Test_Bins;

WITH CTEfill (DueBinID,BinDate,ItemID,Duration) AS (
    -- Anchor
    SELECT --TI.DueBinID, 
           -- (SELECT DATEADD(MINUTE,TI.Duration,MAX(BI.BinDateTime)) FROM Test_BinItems as BI WHERE BI.BinID = TI.DueBinID) as DueBinID, 
           (SELECT MAX(TB1.BinID) FROM Test_Bins as TB1 WHERE TB1.BinID <= TI.DueBinID AND TB1.BinCapacity - (SELECT SUM(TBI.Duration) FROM Test_BinItems as TBI WHERE TBI.BinID = TB1.BinID) > 0) as DueBinID,
           (SELECT MAX(TB1.BinDateTime) FROM Test_Bins as TB1 WHERE TB1.BinID <= TI.DueBinID AND TB1.BinCapacity - (SELECT SUM(TBI.Duration) FROM Test_BinItems as TBI WHERE TBI.BinID = TB1.BinID) > 0) as BinDate,
           TI.ItemID,
           CASE -- Get remaining (usually whole) item duration if enough capacity in selected bin is available, otherwise space left in selected bin
                WHEN TI.Duration - (SELECT SUM(BI1.Duration) FROM Test_BinItems as BI1)
                   < (SELECT BinCapacity FROM Test_Bins as BI2 WHERE BI2.BinID = TI.DueBinID) - (SELECT SUM(Duration) FROM Test_Items as TI2 WHERE TI2.DueBinID = TI.DueBinID) 
                THEN TI.Duration - (SELECT SUM(BI1.Duration) FROM Test_BinItems as BI1)
                ELSE (SELECT BinCapacity FROM Test_Bins as BI2 WHERE BI2.BinID = TI.DueBinID) - (SELECT SUM(Duration) FROM Test_Items as TI2 WHERE TI2.DueBinID = TI.DueBinID)
           END as Duration
    FROM Test_Items as TI

    UNION ALL   -- Add recursion to the initial elements

    SELECT DueBinID - 1, 
           (SELECT BinDate FROM Test_Bins as TB3 WHERE TB3.BinID = CTE.DueBinID - 1), 
           CTE.ItemID,
    --     CASE -- Get remaining (usually whole) item duration if enough capacity in selected bin is available, otherwise space left in selected bin
    --          WHEN (SELECT Duration FROM Test_Items as TI3 WHERE TI3.ItemID = CTE.ItemID) - (SELECT SUM(BI1.Duration) FROM Test_BinItems as BI1)
    --             < (SELECT BinCapacity FROM Test_Bins as BI2 WHERE BI2.BinID = CTE.DueBinID - 1) - (SELECT SUM(Duration) FROM Test_Items as TI2 WHERE TI2.DueBinID = CTE.DueBinID - 1) 
    --          THEN (SELECT Duration FROM Test_Items as TI3 WHERE TI3.ItemID = CTE.ItemID) - (SELECT SUM(BI1.Duration) FROM Test_BinItems as BI1)
    --          ELSE (SELECT BinCapacity FROM Test_Bins as BI2 WHERE BI2.BinID = CTE.DueBinID - 1) - (SELECT SUM(Duration) FROM Test_Items as TI2 WHERE TI2.DueBinID = CTE.DueBinID - 1)
    --     END as Duration
           CTE.Duration
    FROM CTEfill as CTE
    WHERE BinDate > @MinDate AND DueBinID > 0 -- AND Duration > 0           
)

INSERT INTO [dbo].[Test_BinItems]  -- create new item assignment to bin
            ([BinID]
            ,[BinDateTime]
            ,[ItemID]
            ,[Duration])
SELECT DueBinID, BinDate, ItemID, Duration FROM CTEfill;

样本数据

CREATE TABLE Test_Bins(
    BinID INT,
    BinDateTime DateTime, 
    BinCapacity INT
)

-- Sample bins
INSERT INTO Test_Bins(BinID,BinDateTime,BinCapacity) VALUES (1,'2001-01-01 00:00', 9999);    -- **overflow** bin 
INSERT INTO Test_Bins(BinID,BinDateTime,BinCapacity) VALUES (2,'2019-07-15 06:45', 270);     -- new pre-set bin
INSERT INTO Test_Bins(BinID,BinDateTime,BinCapacity) VALUES (3,'2019-07-16 10:00', 540);     -- new pre-set bin
INSERT INTO Test_Bins(BinID,BinDateTime,BinCapacity) VALUES (4,'2019-07-17 00:30', 120);     -- new pre-set bin
INSERT INTO Test_Bins(BinID,BinDateTime,BinCapacity) VALUES (5,'2019-07-19 06:00', 480);     -- new pre-set bin
INSERT INTO Test_Bins(BinID,BinDateTime,BinCapacity) VALUES (6,'2019-07-22 07:15', 270);     -- new pre-set bin


CREATE TABLE Test_Items(
    DueBinID INT,
    ItemID INT,
    DueDate DateTime, 
    Duration INT
);

-- Sample items
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (5,418050,'2019-07-21',23); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (5,418053,'2019-07-19',183); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (5,418066,'2019-07-19',56); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (5,418077,'2019-07-19',32); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (5,418087,'2019-07-19',80); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (5,418091,'2019-07-19',38); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (5,418101,'2019-07-19',101); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (4,418113,'2019-07-18',125); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (4,418116,'2019-07-18',12); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (4,418124,'2019-07-18',6); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (4,418131,'2019-07-18',48); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (4,418134,'2019-07-18',70); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (4,418139,'2019-07-17',20); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (4,418156,'2019-07-17',80); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (4,418158,'2019-07-17',5); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (3,418168,'2019-07-16',18); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (3,418177,'2019-07-16',13); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (3,418179,'2019-07-16',45); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (3,418183,'2019-07-16',20); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (3,418294,'2019-07-16',20); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (2,418296,'2019-07-15',123); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (2,418298,'2019-07-15',242); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (2,418299,'2019-07-15',55); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (2,418301,'2019-07-15',74); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (1,418303,'2019-07-14',145); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (1,418305,'2019-07-13',88); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (1,418313,'2019-07-02',29); 
INSERT INTO Test_Items(DueBinID,ItemID,DueDate,Duration) VALUES (1,418414,'2019-06-21',17); 


CREATE TABLE Test_BinItems(
    BinID INT,
    BinDateTime DateTime, 
    ItemID INT,
    Duration INT
);

带有光标的解决方案

DELETE FROM Test_BinItems   -- clear the assignment table

DECLARE @i INT
DECLARE @iDuration INT
DECLARE @iBinCapacity INT
DECLARE @iDueDate DATE
DECLARE @iItemID INT
DECLARE @iBinID INT
DECLARE @BinAvailableCapacity INT
DECLARE ItemCursor CURSOR FOR SELECT TI.ItemID FROM Test_Items as TI ORDER BY DueDate DESC -- create cursor

SET @i = 0

OPEN ItemCursor
FETCH NEXT FROM ItemCursor INTO @iItemID   -- get first item

PRINT CONCAT('STARTING @iItemID=',@iItemID)

WHILE @@FETCH_STATUS = 0
BEGIN
    SELECT @iDuration = Duration,   -- start with whole duration of an item
           @iDueDate = DueDate,     -- set due date of iterated item
           @iBinID = DueBinID       -- set starting bin as the due bin of iterated item
    FROM Test_Items WHERE ItemID = @iItemID   -- set variables for iterated item
    PRINT CONCAT('iITEM:   @iDuration=',@iDuration,'    @iDueDate=',@iDueDate,'    @iBinID=',@iBinID)

    WHILE @iDuration > 0  AND @i < 500           -- until we allocate whole duration of iterated item to bins (with safety limit of 500 iterations)
    BEGIN
        SET @BinAvailableCapacity = 
              (SELECT BinCapacity FROM Test_Bins WHERE BinID = @iBinID)                   -- whole bin capcacity
            - ISNULL((SELECT SUM(Duration) FROM Test_BinItems WHERE BinID = @iBinID),0)   -- minus already filled capacity
        PRINT CONCAT('INSETING @BinAvailableCapacity=',@BinAvailableCapacity)
        IF @BinAvailableCapacity > 0 -- if there is capacity to be used
        BEGIN
            -- Used duration in this iteration
            DECLARE @iUsedDuration INT
            IF @iDuration < @BinAvailableCapacity  -- if the item fits remaining bin capacity
            BEGIN 
                SET @iUsedDuration = @iDuration
                SET @iDuration = 0
            END
            ELSE  -- if the item does not fit remaining bin capacity, put only part of the item in
            BEGIN
                SET @iUsedDuration = @BinAvailableCapacity
                SET @iDuration = @iDuration - @iUsedDuration
            END

            -- Item date-time assigned in the bin
            DECLARE @iItemBinDateTime DateTime        -- stack date-TIME - where the item will start
            DECLARE @LastBinItemDTInBin DateTime      -- where the last inserted item starts
            SET @LastBinItemDTInBin = (SELECT MAX(BinDateTime) FROM Test_BinItems WHERE BinID = @iBinID)
            SET @iItemBinDateTime = ISNULL(DATEADD(MINUTE,(SELECT Duration FROM Test_BinItems WHERE BinDateTime = @LastBinItemDTInBin),@LastBinItemDTInBin),(SELECT BinDateTime FROM Test_Bins WHERE BinID = @iBinID))

            INSERT INTO [dbo].[Test_BinItems]  -- create new item assignment to bin
                       ([BinID]
                       ,[BinDateTime]
                       ,[ItemID]
                       ,[Duration])
                 VALUES
                       (@iBinID
                       ,@iItemBinDateTime
                       ,@iItemID
                       ,@iUsedDuration)
            PRINT CONCAT('INSETING @iBinID=',@iBinID,'    @iItemBinDateTime=',@iItemBinDateTime,'    @iItemID=',@iItemID,'    @iUsedDuration=',@iUsedDuration)

        END
        ELSE    -- if there's no capacity to be used, move to previous bin
        BEGIN 
            SET @iBinID = @iBinID - 1 
        END
    SET @i = @i + 1
    END

FETCH NEXT FROM ItemCursor INTO @iItemID
END

CLOSE ItemCursor
DEALLOCATE ItemCursor

0 个答案:

没有答案