从CTE更新表中的记录

Eti*_*nne 58 sql t-sql sql-server common-table-expression

我有以下CTE,它将为我提供整个发票的DocTotal.

 ;WITH CTE_DocTotal
 AS
 (
   SELECT SUM(Sale + VAT) AS DocTotal
   FROM PEDI_InvoiceDetail
   GROUP BY InvoiceNumber
 )

UPDATE PEDI_InvoiceDetail
SET DocTotal = CTE_DocTotal.DocTotal
Run Code Online (Sandbox Code Playgroud)

现在有了这个结果,我想在PEDI_InvoiceDetail中输入DocTotal值列.

我知道不会工作,我知道我错过了什么,它是什么?

Gar*_*thD 102

您对CTE所做的更新将级联到源表.

我不得不稍微猜测你的架构,但这样的事情应该有用.

;WITH T AS
(   SELECT  InvoiceNumber, 
            DocTotal, 
            SUM(Sale + VAT) OVER(PARTITION BY InvoiceNumber) AS NewDocTotal
    FROM    PEDI_InvoiceDetail
)
UPDATE  T
SET     DocTotal = NewDocTotal
Run Code Online (Sandbox Code Playgroud)

  • CTE 更新与普通更新没有任何不同。它是原子的,并且将采用相同的表/行锁。无非是语法糖而已 (3认同)

Pau*_*ing 32

WITH CTE_DocTotal (DocTotal, InvoiceNumber)
AS
(
    SELECT  InvoiceNumber,
            SUM(Sale + VAT) AS DocTotal
    FROM    PEDI_InvoiceDetail
    GROUP BY InvoiceNumber
)    
UPDATE PEDI_InvoiceDetail
SET PEDI_InvoiceDetail.DocTotal = CTE_DocTotal.DocTotal
FROM CTE_DocTotal
INNER JOIN PEDI_InvoiceDetail ON ...
Run Code Online (Sandbox Code Playgroud)


pod*_*ska 24

你不需要CTE

UPDATE PEDI_InvoiceDetail
SET
    DocTotal = v.DocTotal
FROM
     PEDI_InvoiceDetail
inner join 
(
   SELECT InvoiceNumber, SUM(Sale + VAT) AS DocTotal
   FROM PEDI_InvoiceDetail
   GROUP BY InvoiceNumber
) v
   ON PEDI_InvoiceDetail.InvoiceNumber = v.InvoiceNumber
Run Code Online (Sandbox Code Playgroud)

  • SQL中的@Zack"正确性"和"可读性"在某种程度上是主观的,也是品味的问题.但是,某些历史版本的SQL Server不包含CTE功能. (4认同)
  • @HolgerJakobs这个问题是关于SQL Server的.SQLite将拥有自己的语法 (4认同)
  • 问题是关于如何从CTE更新记录,而不是替代方法。OP可能希望使用CTE的原因有很多,但问题并未透露。 (3认同)
  • @DerekGreer OP表示“我想在PEDI_InvoiceDetail内的DocTotal值栏中输入”。他没有说自己需要或想要使用CTE。 (2认同)
  • 正确,除了他在标题中总结了他的问题:“从CTE更新表中的记录”。作为证据,我正在阅读OP希望获得涉及CTE的答案的证据,我提供证据表明他接受了使用CTE提供解决方案的答案,并且他在后续评论中表示,他选择了CTE来提高绩效。 (2认同)