繁体   English   中英

根据结果​​集在数据库中更新或插入记录

[英]Update or Insert records in database based on result set

我有这样的桌子

在此处输入图片说明

我还有另一个表sections ,其中有6个节,其ID从1 to 6 我从用户那里获得了部分ID的列表,该列表提供了有关用户当前活动部分的信息。 假设返回给我的列表具有ID为1用户,其区段ID如下{2,3,4,5}所示。 现在我的问题是

  1. 如何将此部分ID列表传递给存储过程
  2. 我要更新的是记录的活动标志,其中段ID为1,用户ID为1,因为段ID的列表中没有条目1
  3. 我想在用户ID 1的表中插入新的ID为5的记录,因为它在ID列表中返回。

谁能告诉我如何实现这一目标?

我可以从以下查询中获取total section id's

select Id from sections

但不知道我会在total list of section id's之间进行迭代,并比较从C#返回list of section idslist of section ids

回答完整的问题。

1.就像我在评论中说的: 表值参数

首先创建一个UDTT来存储您的部分ID,以供存储过程输入。

CREATE TYPE [dbo].[SectionIdUDTT] AS TABLE(
    [id] int NOT NULL
)

将此UDTT用作存储过程的参数:

ALTER PROCEDURE [dbo].[YourSPname] @SectionId SectionIdUDTT readonly,
    @UserId INT
AS
BEGIN

为了调用此存储过程,必须先填充SectionIdUDTT,然后再调用存储过程。

DECLARE @SectionId AS SectionIdUDTT;

INSERT INTO @SectionId 
--your values here

EXEC YourSPname @SectionId, @UserId;

看一下DataTable类,以便从C#调用它。 确保用户具有对UDTT的执行权限。

2在存储过程中

将记录设置为不活动的用户已存在但不在表值参数中。

UPDATE YourTable
SET IsActive = 0
WHERE UserId = @UserId
    AND SectionId NOT IN (SELECT id FROM @SectionId)
    AND SectionId IN (SELECT Id FROM sections) --Not really needed

3在存储过程中

只需插入尚不存在的记录。 我只是假设id是一个身份列。

INSERT INTO YourTable (UserId, SectionId, IsActive)
SELECT @UserId,
    s.id,
    1
FROM @SectionId s
WHERE NOT EXISTS (
    SELECT 1
    FROM YourTable y
    WHERE y.SectionId = s.id
        AND y.UserId = @UserId
    )

我建议您在存储过程中进行事务管理

首先,我创建了一个将调用存储过程的方法,向该方法传递了一个sections列表(节表实体)和客户ID

public bool SaveChatSectionUserMapping(List<Sections> lstSections, int customerId)
        {
            con = new SqlConnection(connectionString);
            bool isUpdated = false;
            try
            {
                string xmlString = string.Empty;
                xmlString = XMLOperations.WriteXML(lstSections);
                SqlCommand cmd = new SqlCommand("spUpdateSections", con);
                cmd.CommandType = CommandType.StoredProcedure;
                cmd.Parameters.Add("@XMLData", SqlDbType.Xml).Value = xmlString;
                cmd.Parameters.Add("@CustomerId", SqlDbType.Int).Value = customerId;
                SqlParameter param = new SqlParameter();
                param.SqlDbType = SqlDbType.Bit;
                param.Direction = ParameterDirection.Output;
                param.ParameterName = "@Result";
                cmd.Parameters.Add(param);
                con.Open();
                cmd.ExecuteNonQuery();
                isUpdated = (param.Value != DBNull.Value) ? Convert.ToBoolean(param.Value) : false;
            }
            catch (Exception ex)
            {
                throw ex;
            }
            finally
            {
                if (con.State == ConnectionState.Open)
                    con.Close();
            }
            return isUpdated;
        }

我从这一行获得的xmlString的值xmlString = XMLOperations.WriteXML(lstSections); 就是这样

<ArrayOfSection xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Sections>
    <UserId>1</UserId>
    <SectionId>1</SectionId>
    <IsActive>true</IsActive>
  </Sections>
  <Sections>
    <UserId>1</UserId>
    <SectionId>2</SectionId>
    <IsActive>true</IsActive>
  </Sections>
  <Sections>
    <UserId>1</UserId>
    <SectionId>5</SectionId>
    <IsActive>true</IsActive>
  </Sections>
</ArrayOfSection>

现在在存储过程中

CREATE Procedure [dbo].[spUpdateSections]  
(  
 @XMLData as XML,
 @CustomerId INT,  
 @Result int Output

)  
AS
BEGIN
SET NOCOUNT ON;  
    Declare @ErrorCode Varchar(100) = '',@propertyCount VArchar(100) = '',@currentCount int=1,@SectionId int, @IsActive bit

Begin TRY

UPDATE Sections
SET
    IsActive = 0
WHERE
    UserId = @CustomerId

SELECT @propertyCount = convert(VARCHAR, @XMLData.query ('count(/ArrayOfSection/Sections)'))
SET @currentCount = 1

     while (@currentCount<=@propertyCount)       
     Begin 
     SET @SectionId = @XMLData.value('data(/ArrayOfSection/Sections[sql:variable("@currentCount")]/SectionId)[1]', 'INT') 
     SET @IsActive = @XMLData.value('data(/ArrayOfSection/Sections[sql:variable("@currentCount")]/IsActive)[1]', 'BIT')

               If Exists (SELECT *
                          FROM
                              Sections
                          WHERE
                              UserId = @CustomerId
                              AND SectionId = @SectionId)
BEGIN
       IF(@IsActive=1) 
       BEGIN 
            UPDATE Sections
             SET
                 IsActive = 1
             WHERE
                 UserId = @CustomerId AND
                 SectionId = @SectionId

        END
END

ELSE
       BEGIN
       IF(@IsActive=1) 
       BEGIN 
INSERT INTO Sections
           ([SectionId]
           ,[UserId]
           ,[IsActive])
     VALUES
           (@SectionId,@CustomerId,1)
        END
    END

SET @currentCount = @currentCount + 1 
       End 

       SET @Result = 1
           ErrorCode:  
           If(@ErrorCode !='')
           BEGIN
--SELECT @BSErrorResult = doctor.GetErrorCodeDetail(@ErrorCode)
SET @Result = 2
           END  
    END TRY        
    BEGIN CATCH     
      --Declaring Variable for formating error  
      Declare @ErrorMessage  VARCHAR(max),  
       @ErrorSeverity  INT,  
       @ErrorState   INT

--SELECTING TECHNICAL ERROR   
SELECT @ErrorMessage = error_message()
     , @ErrorSeverity = error_severity()
     , @ErrorState = error_state()
     , @ErrorMessage = @ErrorMessage + '  ' + db_name() + '  ' + ' ' + object_name(@@PROCID);   

      RAISERROR (  
         @ErrorMessage, -- Message text.  
         @ErrorSeverity, -- Severity.  
         @ErrorState -- State.  
           );   
     End CATCH    



END

生成XMl的另一种方法是使用XElement这样的

XElement xml = new XElement("Sections",
                                from col in lstSections
                                select new XElement("rows",
                                               new XElement("UserId", col.UserId),
                                               new XElement("SectionId", col.SectionId),
                                               new XElement("IsActive", col.IsActive)));

                              string  xmlString = xml.ToString();

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM