我们的项目中有 2 个表,WorkOrders 和 WorkOrderTrackings,WorkOrderTrackings 有一个外键 WorkOrderId,引用 WorkOrders 中的 Id(这样您就知道跟踪记录属于哪个工作订单)。 WorkOrderTrackings 中的记录代表工作订单从开始到结束所经历的“步骤”。一旦我们的工程部门将工单发布到工厂车间进行 build ,它们就会被创建。因此任何工作订单都可以有一对多的工作订单跟踪。
现在,WorkOrderTrackings 有一个 StageNum 字段,该字段按从开始到结束的顺序说明步骤的步骤号。有日期时间吗?开始日期和日期时间?完成日期字段。当他们开始一个步骤时,开始日期被设置,当他们完成时,完成日期被设置。当完成1步,还没有开始下一步时,上一步的completedate有值,但下一步的startdate没有值。
目前,为了确定工单 X 处于哪个步骤,我们使用 linq 表达式:
var trackingsForX=db.WorkOrderTrackings.Where(wot=>wot.WorkOrderId==X.Id);
var currentTrackingForX=trackingsForX.OrderByDescending(t=>t.StageNum)
.Where(t=>t.StartDate.HasValue && !t.CompleteDate.HasValue)
.FirstOrDefault() ??
trackingsForX.OrderBy(t=>t.StageName)
.Where(t=>!t.CompleteDate.HasValue)
.FirstOrDefault()
当搜索从最后一个到第一个 stagenum 的跟踪时,通过返回找到的带有 startdate 值但没有完成的第一个跟踪记录来选择工作订单的最新跟踪步骤,或者,如果返回 null,则通过查找第一个记录从第一步到最后一步搜索没有完成值。
我们不想每次都这样做,因此我们向 WorkOrders 添加一个字段 CurrentTrackingStepId,该字段将作为工作订单最新跟踪记录的外键。当新步骤开始时,我们将更新工作订单的 CurrentTrackingStepId 以指向该步骤。这样,我们只需查找工单即可随时访问工单的当前步骤。但是,现在我需要制作一个可以运行的 SQL 脚本,它将自动将所有现有工作订单的 CurrentTrackingStepId 设置为该订单的当前跟踪 ID。
如何编写 SQL 脚本来完成此任务?
编辑:使用示例数据库数据更新
dbo.WorkOrderTrackings:
ID WO-Id StageNum StartDate CompleteDate
551874 53781 1 2020-08-28 5:13:34 PM 2020-08-28 5:13:45 PM
551875 53781 2 2020-08-28 7:01:35 PM 2020-08-28 7:01:35 PM
551878 53781 5 2020-08-28 7:01:35 PM NULL
551879 53781 6 NULL NULL
551880 53781 7 NULL NULL
551881 53781 8 NULL NULL
551882 53781 9 NULL NULL
551883 53781 10 NULL NULL
551884 53781 11 NULL NULL
551885 53781 12 NULL NULL
551886 53781 13 NULL NULL
551887 53781 14 NULL NULL
551888 53781 15 NULL NULL
551889 53781 16 NULL NULL
551890 53781 17 NULL NULL
551891 53803 1 2020-08-28 6:33:18 PM 2020-08-28 6:33:18 PM
551892 53803 2 2020-08-28 6:33:18 PM 2020-08-31 8:07:09 AM
551893 53803 3 2020-08-31 8:07:17 AM 2020-08-31 8:07:24 AM
551894 53803 4 NULL NULL
551895 53803 5 NULL NULL
551896 53803 6 NULL NULL
551897 53803 7 NULL NULL
551908 53803 18 NULL NULL
551909 53803 19 NULL NULL
551910 53803 20 NULL NULL
551911 53803 21 NULL NULL
551912 53803 22 NULL NULL
551913 53803 23 NULL NULL
551914 53803 24 NULL NULL
551915 53803 25 NULL NULL
551916 53803 26 NULL NULL
dbo.WorkOrders:
Id OrderNumber
53803 139107-00
53781 139094-00
因此,如果您按照上面的数据,如果我想要订单 139107-00 的当前跟踪步骤,它应该返回 ID 为 551894 的 WorkOrderTracking 记录(因为这是紧接在最近完成的记录之后的记录)。 如果我想要订单 139094-00 的当前跟踪步骤,它应该返回 ID 为 551878 的 WorkOrderTracking 记录(因为这是它已开始但尚未完成的步骤)。
因此,根据上面的数据,我想将 WorkOrders 中的新 CurrentTrackingStepId 字段设置为 WorkOrderTrackings 中具有 StartDate 值但没有 CompleteDate 值的订单的最新跟踪记录的 ID,或者,如果返回 null ,我想要最后一个具有 StartDate 值和 CompleteDate 值的记录之后的下一个 StageNum 的记录 ID。这有道理吗?
我确实没有尝试过任何脚本,因为我不知道如何编写它。我意识到我需要 dbo.WorkOrders 的更新来设置 CurrentTrackingStepId=...某事。这需要有一个 select 语句来从 WorkOrderTrackings 表中获取正确记录的 Id。它可能需要一个 EXIST 语句来检查是否存在包含开始日期但没有完整日期的记录。
最佳答案
假设您的 StageNum
值是连续的,您可以执行以下操作:
/* Table and data mock-up */
DECLARE @WorkOrders table ( Id int, OrderNumber varchar(10), CurrentStage int );
INSERT INTO @WorkOrders ( Id, OrderNumber ) VALUES
( 53803, '139107-00' ), ( 53781, '139094-00' );
DECLARE @WorkOrderTrackings table (
ID int, [WO-id] int, StageNum int, StartDate datetime, CompleteDate datetime
);
INSERT INTO @WorkOrderTrackings VALUES
( 551891, 53803, 1, '2020-08-28 6:33:18 PM', '2020-08-28 6:33:18 PM' ),
( 551892, 53803, 2, '2020-08-28 6:33:18 PM', '2020-08-31 8:07:09 AM' ),
( 551893, 53803, 3, '2020-08-31 8:07:17 AM', '2020-08-31 8:07:24 AM' ),
( 551894, 53803, 4, NULL, NULL ),
( 551895, 53803, 5, NULL, NULL ),
( 551896, 53803, 6, NULL, NULL ),
( 551897, 53803, 7, NULL, NULL );
/* Update the current stage of the WorkOrder based on the last completed StageNum + 1 */
UPDATE @WorkOrders
SET
CurrentStage = wt.CurrentStageID
FROM @WorkOrders wo
OUTER APPLY (
SELECT ID AS CurrentStageID FROM @WorkOrderTrackings wt
WHERE
wt.[WO-id] = wo.Id AND wt.StageNum = (
SELECT ( MAX( StageNum ) + 1 ) FROM @WorkOrderTrackings x WHERE x.[WO-id] = wt.[WO-id]
AND x.CompleteDate IS NOT NULL
)
) AS wt
WHERE
wo.[OrderNumber] = '139107-00';
/* Show the updated WorkOrders resultset */
SELECT * FROM @WorkOrders ORDER BY Id;
返回
+-------+-------------+--------------+
| Id | OrderNumber | CurrentStage |
+-------+-------------+--------------+
| 53781 | 139094-00 | NULL |
| 53803 | 139107-00 | 551894 |
+-------+-------------+--------------+
更新:
I like the approach, but notice that
StageNum
isn't contiguous (1,2,5,6,7 in the first series).
此示例使用 Gert 方法中的 LEAD。
/* Update the current stage of the WorkOrder based on the last stage completed */
UPDATE @WorkOrders
SET
CurrentStage = wt.CurrentStageID
FROM @WorkOrders wo
OUTER APPLY (
SELECT TOP 1
LEAD( ID, 1, NULL ) OVER ( PARTITION BY [WO-id] ORDER BY ID, StageNum ) AS CurrentStageID
FROM @WorkOrderTrackings wt
WHERE
wt.[WO-id] = wo.Id AND wt.ID >= (
SELECT MAX( ID ) FROM @WorkOrderTrackings x WHERE x.[WO-id] = wt.[WO-id]
AND x.CompleteDate IS NOT NULL
)
ORDER BY
wt.ID, wt.StageNum
) AS wt
WHERE
wo.[OrderNumber] = '139107-00';
返回
+-------+-------------+--------------+
| Id | OrderNumber | CurrentStage |
+-------+-------------+--------------+
| 53781 | 139094-00 | NULL |
| 53803 | 139107-00 | 551894 |
+-------+-------------+--------------+
关于sql - 如何使用 Sql 服务器中另一个表中的值的条件来更新表中的值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64932611/