我正在使用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?
时无法弄清楚...... 帮助确实很感激!
答案 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中使用以下属性设置该字段:
最后一个确保如果在数据表中插入新值,新ID值将不会与现有值相冲突