0

大家晚上好!

我遇到了一个我无法理解的非常奇怪的问题。

我有 3 个表(零件表、零件移动历史记录和零件详细信息表)。

我要做的是让结果集返回批次#、零件#、产品描述、数量、零件位置、当前库存(相对于完整历史记录)以及上次移动产品的人。

现在,对于查询。当我运行以下查询时,我得到 4,751 行的结果集;这完全符合我的预期结果。但是,当我尝试添加用户 ID 字段时,我得到的结果集为 186,573。这个大型结果集似乎提取了所有历史数据,而不是仅将用户 ID 与我实际需要的 4,751 行匹配。

从我需要的部件表 (prod_desc) 从我需要的部件详细信息表 (lot,part#,lotquantity,prtlocation) 从我需要的部件移动历史表 (move_date,user_id)

4,751 查询:

SELECT DISTINCT
inv.lot,
inv.part#,
prt.prod_desc,
inv.lotquantity,
inv.prtlocation,
MAX(mv.move_date)AS 'Move Date'
FROM invdet AS inv
LEFT JOIN movetable AS mv ON inv.part# = mv.part#
LEFT JOIN partmstr AS prt ON inv.part# = prt.part#
WHERE inv.lot IS NOT NULL
GROUP BY inv.lot,inv.part#,prt.prod_desc,inv.lotquantity,inv.prtlocation
ORDER BY inv.prtlocation

186,573 查询:

SELECT DISTINCT
inv.lot,
inv.part#,
prt.prod_desc,
inv.lotquantity,
inv.prtlocation,
MAX(mv.move_date)AS 'Move Date'
mv.user_id
FROM invdet AS inv
LEFT JOIN movetable AS mv ON inv.part# = mv.part#
LEFT JOIN partmstr AS prt ON inv.part# = prt.part#
WHERE inv.lot IS NOT NULL
GROUP BY inv.lot,inv.part#,prt.prod_desc,inv.lotquantity,inv.prtlocation,mv.user_id
ORDER BY inv.prtlocation

如果我不使用 MAX 函数,我不会获得当前库存,而是会在表中获得所有结果,而我不需要。我还在学习,我的 GROUP BY 有很多不足之处,因为我还在纠结它(欢迎提出建议!)。我敢肯定有一个子查询我可以在这里的某个地方抛出,但我仍在弄清楚这些。任何帮助是极大的赞赏!

4

1 回答 1

0

我认为问题在于,当您从表可移动表中插入 mv.user_id 时,您会得到所有部分的运动,而不仅仅是日期为 max(mv.move_date) 的最后一个。一种方法是删除左连接到可移动并使用交叉应用,如

SELECT inv.lot,inv.part,prt.prod_desc,inv.lotquantity,inv.prtlocation,x.move_date,x.user_id
FROM invdet AS inv
CROSS APPLY(SELECT TOP 1
             mv.user_id,mv.move_date
            FROM movetable mv
            WHERE inv.part=mv.part
            ORDER BY mv.move_date DESC) AS x
LEFT JOIN partmstr AS prt ON inv.part=prt.part
WHERE inv.lot IS NOT NULL
ORDER BY inv.prtlocation

我没有测试过,但应该没问题,可能有点慢,因为交叉应用对 inv 表中的每一行执行一个子查询。如果速度太慢,可以用 ROWNUMBER 创建一个只包含最后一个动作的表,然后在 LEFT JOIN 中使用,如下

SELECT inv.lot,inv.part,prt.prod_desc,inv.lotquantity,inv.prtlocation,y.move_date,y.user_id
FROM invdet AS inv
LEFT JOIN(SELECT x.user_id,x.move_date,x.part
          FROM (SELECT mv.user_id,mv.move_date,mv.part,rn=ROWNUMBER() OVER(PARTITION BY mv.part ORDER BY mv.move_date DESC)
                FROM movetable mv) AS x
          WHERE x.rn=1) AS y ON y.part=inv.part
LEFT JOIN partmstr AS prt ON inv.part=prt.part
WHERE inv.lot IS NOT NULL
ORDER BY inv.prtlocation

希望能帮助到你。

于 2019-03-02T22:15:42.803 回答