开发者

LINQ Insert Into Database resulted in duplicates

开发者 https://www.devze.com 2023-01-20 12:01 出处:网络
I have a linq query running in a WCF Web Service that looks for a match and if one is not found then it creates one.

I have a linq query running in a WCF Web Service that looks for a match and if one is not found then it creates one.

my code looks like

   //ReadComm开发者_Go百科itted transaction
   using (var ts = CreateTransactionScope(TransactionScopeOption.RequiresNew))
   {
    Contract contract = db.Contracts.SingleOrDefault(x => x.txtBlah == str);
    if (contract == null)
    {
      contract = new Contract();
      contract.txtBlah = str;
      db.Contracts.InsertOnSubmit(contract);
      db.SubmitChanges();
    }
    ...
    db.SubmitChanges();
   }

The problem is that I am getting duplicates. I thought the transaction would have locked the database to ensure no duplicates would be found (and supply the ability to rollback). How can I ensure that there are no duplicates?


In a ReadCommited transaction the data can be changed before the end of the transaction. But you can use Serializable transaction which will do table locking.

db.Connection.Open();
using (db.Transaction = db.Connection.BeginTransaction(IsolationLevel.Serializable))
{
//your code here
db.Transaction.Commit();
}

Ok, if you don't want to use Serializable tran then you should write a sproc for atomic insert, the logic should look like this SQL in code:

db.ExecuteCommand("INSERT INTO Contract (txtBlah) SELECT {0} WHERE NOT EXISTS (SELECT 1 FROM Contract WITH (TABLOCK) WHERE txtBlah={0})", str);

note this will also lock the whole table during the insert.

Read more on how to create a sproc without a race condition at http://weblogs.sqlteam.com/dang/archive/2007/10/28/Conditional-INSERTUPDATE-Race-Condition.aspx.

0

精彩评论

暂无评论...
验证码 换一张
取 消