【问题标题】:Insert or Update or Delete Data From Table Value Parameter从表值参数中插入、更新或删除数据
【发布时间】:2014-05-14 15:46:29
【问题描述】:

我有一个 ProductsAttributesValues 表,

Id   ProductId   AttributeId   Value   ORDER
-------------------------------------------------

具有属性(相同 ProductId/AttributeId)的产品可以有多个值但具有不同的顺序。我有一个类型,

CREATE TYPE [dbo].[ProductsAttributesValuesType] AS TABLE
(
    [AttributeId] [INT],
    [Value] [NVARCHAR](MAX)
)

我在我的存储过程中将这个ProductsAttributesValuesType 作为带有 ProductId 参数的表值参数传递。现在我需要的东西似乎很简单,但我被卡住了。 ProductsAttributesValuesType 中的信息来自 UI。所以,

1) 如果值为 null、空或空白,则我无需执行任何操作并从 ProductsAttributesValues 中删除所有具有相同 ProductId/AttributeId 的现有记录。

2) 如果一个值不为空、非空或非空白,则可能存在单个值或多个值(ProductsAttributesValuesType 中的多个记录)。如果有单个值,如果不存在现有值,我需要插入,如果存在值则更新。如果有多个值(ProductsAttributesValuesType 中的多个记录),如果不存在现有值,我需要插入所有值,如果存在值,则需要更新/插入。

一种方法是简单地删除所有具有相同 ProductId/AttributeId 的 ProductsAttributesValues 表。然后插入具有非空值的值。但是我觉得效率不是很高。

更新:目前,我首先删除所有多值属性值。接下来,我将删除值为空的所有属性值。接下来,我将插入/更新单值属性值。接下来,我将插入多值属性值。

【问题讨论】:

  • 我知道 MERGE,但我被卡住了。你能解决这个问题吗?
  • 包括您迄今为止尝试过的内容以及一组源/结果示例。
  • 我认为您在上一段中描述的方法可能会导致最干净的代码。我不会担心性能。一些更新与删除/插入之间的区别不会那么大。
  • @GordonLinoff,感谢您的专家意见

标签: sql sql-server sql-server-2012


【解决方案1】:

避免使用 Merge 语句将其分解为三个语句并将其全部包装在一个事务中,如下所示

CREATE  Procedure Upserting_To_ProductsAttributesValues 
@ProductsAttributesValuesType   [dbo].[ProductsAttributesValuesType] READONLY
AS 
BEGIN
 SET NOCOUNT ON;
 BEGIN TRY
    BEGIN TRANSACTION;

-- Delete values 
  DELETE FROM ProductsAttributesValues 
  FROM ProductsAttributesValues T INNER JOIN @ProductsAttributesValuesType P
  ON T.AttributeId = P.AttributeId
  WHERE P.Value IS NULL OR RTRIM(LTRIM(P.Value)) = ''

-- Update Statement 
UPDATE T
SET T.Value = P.Value
FROM ProductsAttributesValues T INNER JOIN @ProductsAttributesValuesType P
ON T.AttributeId = P.AttributeId

-- Insert Statement
INSERT INTO ProductsAttributesValues (AttributeId, Value)
SELECT AttributeId, Value
FROM @ProductsAttributesValuesType P
WHERE NOT EXISTS (SELECT 1 
                  FROM ProductsAttributesValues
                  WHERE AttributeId = P.AttributeId)

    COMMIT TRANSACTION;  --<-- If nothing went wrong
 END TRY

 BEGIN CATCH


     -- Now make use of ERROR Functions to get detailed information
     -- about the error, these functions are only allowed in catch block

 SELECT ERROR_MESSAGE() AS [ERROR_MESSAGE]
       ,ERROR_LINE()    AS [ERROR_LINE]
       ,ERROR_NUMBER()  AS [ERROR_NUMBER] 
 END CATCH

END

注意

我建议避免使用 Merge 语句的原因请参阅这篇文章作者 Aaron Bertrand Use Caution with SQL Server's MERGE Statement

【讨论】:

  • 感谢阿里,但我有多个值。
  • 这应该可以,但是在 DELETE 部分,你想说的是 LEFT JOIN 而不是 INNER JOIN
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2013-09-22
  • 2012-03-18
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多