sql - 如何在 UPDATE 中避免 GROUP BY 或 DISTINCT
问题描述
我有一张这样的桌子
A B C D E F
00002471 Sd3a28d471 0 24.00 377.500000 1
00002471 Sd3a28d471 0 353.50 377.500000 1
00002471 Sd3a28d471 1 211.00 211.000000 1
00002471 Sd3a28e471 1 343.00 343.000000 1
00002471 Sd3a28e471 0 56.00 242.370000 1
00002471 Sd3a28e471 0 177.06 242.370000 1
00002471 Sd3a28e471 0 9.31 242.370000 1
00002471 Sd3a28f471 0 10.31 10.31 1
00002471 Sd3a28f471 1 10.31 10.31 1
通过对 A、B 和 C 列进行分组,我需要检查 E 中哪一个具有最低值,并将最低值行的 F 列更新为 1,将其余列更新为 0。如果值相同,我需要将 F 列更新为 1,其中 C 为 1。我需要的输出如下
A B C D E F
00002471 Sd3a28d471 0 24.00 377.500000 0
00002471 Sd3a28d471 0 353.50 377.500000 0
00002471 Sd3a28d471 1 211.00 211.000000 1
00002471 Sd3a28e471 1 343.00 343.000000 0
00002471 Sd3a28e471 0 56.00 242.370000 1
00002471 Sd3a28e471 0 177.06 242.370000 1
00002471 Sd3a28e471 0 9.31 242.370000 1
00002471 Sd3a28f471 0 10.31 10.31 0
00002471 Sd3a28f471 1 10.31 10.31 1
我在下面的查询中试过这个
UPDATE T1
SET T1.F = CASE WHEN T1.E <= T2.E THEN 1 ELSE 0 END
--select t2.*
FROM
(SELECT DISTINCT A,B,C,D,E,F FROM #SalesOrder WHERE E IS NOT NULL) T1
INNER JOIN
(SELECT DISTINCT A,B,C,D,E,F FROM #SalesOrder WHERE E IS NOT NULL) T2
ON T1.A = T2.A
AND T1.B = T2.B
--AND T1.C = T2.C
WHERE T1.C = 1 AND T2.C = 0
但是此查询会引发错误
错误消息
消息 4418,级别 16,状态 1,行 265
派生表“T1”不可更新,因为它包含聚合、DISTINCT 或 GROUP BY 子句或 PIVOT 或 UNPIVOT 运算符。
@Suresh Gajera我需要取A,B,C和E的不同值,在取不同的值之后,它会像这样
A B C E
00002471 Sd3a28d471 0 377.500000
00002471 Sd3a28d471 1 211.000000
00002471 Sd3a28e471 0 242.370000
00002471 Sd3a28e471 1 343.000000
00002471 Sd3a28f471 0 10.31
00002471 Sd3a28f471 1 10.31
在此之后,我需要根据 C 列比较 E 的值,在此第二行与第一行比较时最低,所以 2 行应该是 1,第一个应该是 0 ......以同样的方式第三和第四行应该更新..当涉及到第 5 行和第 6 行时,在这种情况下值相同,我需要将 F 列更新为 1,其中 C 为 1
解决方案
仅当您按 A 和 B(没有 C)分组时,您的预期结果才有意义。
在这种情况下:
update t
set t.f = case when t.e = tt.mine then 1 else 0 end
from tablename t inner join (
select a, b, min(e) mine
from tablename
group by a, b
) tt
on tt.a = t.a and tt.b = t.b
请参阅演示。
或使用CTE
和窗口功能MIN()
:
with cte as (
select *, min(e) over (partition by a, b) mine
from tablename
)
update cte
set f = case when e = mine then 1 else 0 end
请参阅演示。
结果:
> A | B | C | D | E | F
> ---: | :--------- | -: | --: | :----- | -:
> 2471 | Sd3a28d471 | 0 | 24 | 377.50 | 0
> 2471 | Sd3a28d471 | 0 | 353 | 377.50 | 0
> 2471 | Sd3a28d471 | 1 | 211 | 211.00 | 1
> 2471 | Sd3a28e471 | 1 | 343 | 343.00 | 0
> 2471 | Sd3a28e471 | 0 | 56 | 242.37 | 1
> 2471 | Sd3a28e471 | 0 | 177 | 242.37 | 1
> 2471 | Sd3a28e471 | 0 | 9 | 242.37 | 1
推荐阅读
- c# - 无法将数据从控制器传递到报表查看器
- javascript - 从对象 [1] 获取元素并将其全部放在对象 [0] javascript
- python-3.x - 从python列表打印元素时获取错误列表索引超出范围
- java - 为什么我不需要在方法 [Java] 中声明异常?
- hyperledger-fabric - 安装时 Hyperledger Fabric 链码抛出错误
- awk - 当其连续行在同一列中具有负值时,提取在特定列中具有正值的行
- java - 如何在 Jersey 2 的 postfilter 中修改查询参数
- java - 对象的Java数组列表如何搜索元素错误
- python - 启动 pybuilder 项目后,我收到可执行文件无法运行且 virtualenv 不兼容的错误
- c - C中的快速排序为什么要额外交换?