sql - 如何使用 Sql 服务器中另一个表中的值的条件来更新表中的值

标签 sql sql-server asp.net-mvc entity-framework

我们的项目中有 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/

相关文章:

SQL Server通过脚本管理列描述

sql-server - 查找所有正在使用链接服务器的 View

asp.net-mvc - 使用模型绑定(bind)在 MVC3 中上传多个文件

c# - ASP.NET MVC 子目录

mysql - 慢sql查询优化,

c# - 如何通过 LINQ to SQL 返回可变列长度

mysql if 语句 else 返回下一个

mysql - Spring 模板: how to execute multiple SQL statements?

sql-server - SQL Server Management Studio 的昂贵查询中的 Execution/Min 是执行次数还是只是估计值?

.net - 将Kibana仪表板集成到.Net MVC应用程序中?