SqlDataAdapter不更新数据库

时间:2016-02-26 13:29:06

标签: c# .net sql-server ado.net

我正在使用SqlDataAdapter来更新我在数据库中以另一种方法查询的dataTable,并且在该方法中有更新的列,Status和Is_processed。 现在,我希望在数据库中保留(更新)这些更改。以下是为实现目标所做的工作:

代码

 batchSize = 10;

string cmd;

    int updatedRows;
    try
    {
        string connString = db.ConnectionString;
        using (SqlConnection conn = new SqlConnection(connString))
        using (SqlDataAdapter adapter = new SqlDataAdapter())
        {
            cmd = "UPDATE IMPORTED_ACCOUNTS SET STATUS = @Status , IS_PROCESSED = @IsProcessed " +
                   ", CREATED_ON = @CreatedOn , CREATED_BY = @CreatedBy , UPDATED_ON = @UpdatedOn , UPDATED_BY = @UpdatedBy " +


 "WHERE CONVENTIONAL_ACCOUNT = @ConAcct";
        adapter.UpdateCommand = new SqlCommand(cmd, conn);
        adapter.UpdateCommand.UpdatedRowSource = UpdateRowSource.OutputParameters;
        adapter.UpdateCommand.Parameters.AddWithValue("@Status", "Status");
        adapter.UpdateCommand.Parameters.Add("@IsProcessed", SqlDbType.Bit, 1, dTable.Columns["IsProcessed"].ColumnName);
        adapter.UpdateCommand.Parameters.Add("@CreatedOn",SqlDbType.DateTime,30, dTable.Columns["CreatedOn"].ColumnName);
        adapter.UpdateCommand.Parameters.AddWithValue("@CreatedBy", dTable.Columns["CreatedBy"].ColumnName);
        adapter.UpdateCommand.Parameters.Add("@UpdatedOn",SqlDbType.DateTime,30, dTable.Columns["UpdatedOn"].ColumnName);
        adapter.UpdateCommand.Parameters.AddWithValue("@UpdatedBy", dTable.Columns["UpdatedBy"].ColumnName);
        adapter.UpdateCommand.Parameters.AddWithValue("@ConAcct", dTable.Columns["ConventionalAccount"].ColumnName);

        cmd = "INSERT INTO IMPORTED_ACCOUNTS ([STATUS],[IS_PROCESSED],[CREATED_ON],[CREATED_BY],[UPDATED_ON],[UPDATED_BY], CONVENTIONAL_ACCOUNT) " +
              "VALUES (@Status , @IsProcessed, @CreatedOn, @CreatedBy, @UpdatedOn, @UpdatedBy, @ConAcct) ";
        adapter.InsertCommand = new SqlCommand(cmd, conn);
        adapter.InsertCommand.UpdatedRowSource = UpdateRowSource.OutputParameters;
        adapter.InsertCommand.Parameters.AddWithValue("@Status", dTable.Columns["Status"].ColumnName);
        adapter.InsertCommand.Parameters.Add("@IsProcessed", SqlDbType.Bit, 1, dTable.Columns["IsProcessed"].ColumnName);
        adapter.InsertCommand.Parameters.Add("@CreatedOn", SqlDbType.DateTime, 30, dTable.Columns["CreatedOn"].ColumnName);
        adapter.InsertCommand.Parameters.AddWithValue("@CreatedBy", dTable.Columns["CreatedBy"].ColumnName);
        adapter.InsertCommand.Parameters.Add("@UpdatedOn", SqlDbType.DateTime, 30, dTable.Columns["UpdatedOn"].ColumnName);
        adapter.InsertCommand.Parameters.AddWithValue("@UpdatedBy", dTable.Columns["UpdatedBy"].ColumnName);
        adapter.InsertCommand.Parameters.Add("@ConAcct", SqlDbType.VarChar, 100, dTable.Columns["ConventionalAccount"].ColumnName);

        cmd = "DELETE FROM IMPORTED_ACCOUNTS WHERE CONVENTIONAL_ACCOUNT = @ConAcct";
        adapter.DeleteCommand = new SqlCommand(cmd, conn);
        adapter.DeleteCommand.UpdatedRowSource = UpdateRowSource.OutputParameters;
        adapter.DeleteCommand.Parameters.AddWithValue("@ConAcct", dTable.Columns["ConventionalAccount"].ColumnName);
        adapter.UpdateBatchSize = batchSize;
        updatedRows = adapter.Update(dTable);  // point where code breaks
    }
    return updatedRows;
}
catch (Exception ex)
{
    return 0;
}

错误

最终在catch块中出现以下错误:

  

违反PRIMARY KEY约束'PK_ACCTS'。无法插入   对象'dbo.IMPORTED_ACCOUNTS'中的重复键。

注释 为什么在数据库中已经存在时应该更新数据库中的行? adapter.Update(dTable)方法有什么条件,当它触发Update而不是Insert?

无法弄清楚...... 帮助确实很感激!

2 个答案:

答案 0 :(得分:2)

在.NET中DataAdapter根据DataRow.RowState属性进行数据库更改。

如果值为Added,dataadapter将尝试插入记录。 如果您需要更新记录 - 那么此属性应具有Modified值。

您无法直接设置值RowState。例如,当您向dataTable添加行时,它已自动设置为Added

Hovewer,您仍然可以间接修改此值。要执行此操作,您应该为需要更新的thouse datarows调用DataRow.AcceptChanges,这会将RowState设置为Unchanged。然后你应该修改这些数据行中的任何内容,在这种情况下,它们将有RowState = Modified

答案 1 :(得分:1)

适配器将根据表中每行的DataRow.RowState决定何时插入,更新或删除。 有关可能的状态,请参阅https://msdn.microsoft.com/es-es/library/system.data.datarowstate(v=vs.110).aspx

填写DataTable后,您可以调用DataTable.AcceptChanges方法将所有行标记为未更改。

此外,如果您有自动数字PK,最好在DataTable中使用以下属性设置该字段:

  • 自动增量:真实
  • AutoIncrementSeed:-1

最后一个确保如果在数据表中插入新值,新ID值将不会与现有值相冲突