LINQ to SQL语句(13)之开放式并发控制和事务

30 篇文章 0 订阅

转载地址: http://www.prg-cn.com/article-4450-1.html

Simultaneous Changes开放式并发控制

下表介绍 LINQ to SQL 文档 中涉及开放式并发的术语:

术语说明
并发

两个或更多用户同时尝试更新同一数据库行的情形。

并发冲突两个或更多用户同时尝试向 一行的一列或多列提交冲突值的情形。
并发控制用于解决并发冲突的技术。
开放式并发控制先调查其他事务是否已更改了行中的值,再允许提交更改的技术。相 比之下,保守式并发控制则是通过锁定记录来避免发生并发冲突。之所以称作开 放式控制,是因为它将一个事务干扰另一事务视为不太可能发生。
冲突解决通过重新查询数据库刷新出现冲突的项 ,然后协调差异的过程。刷新对象时,LINQ to SQL 更改跟踪器会保留以下数据 :最初从数据库获取并用于更新检查的值 通过后续查询获得的新数据库值。 LINQ to SQL 随后会确定相应对象是否发生冲突(即它的一个或多个成员值是否 已发生更改)。如果此对象发生冲突,LINQ to SQL 下一步会确定它的哪些成员 发生冲突。LINQ to SQL 发现的任何成员冲突都会添加到冲突列表中。

在 LINQ to SQL 对象模型中,当以下两个条件都得到满 足时,就会发生“开放式并发冲突”:客户端尝试向数据库提交更改 ;数据库中的一个或多个更新检查值自客户端上次读取它们以来已得到更新。 此冲突的解决过程包括查明对象的哪些成员发生冲突,然后决定您希望如何进行 处理。

开放式并发(Optimistic Concurrency)

说明:这个例子中在 你读取数据之前,另外一个用户已经修改并提交更新了这个数据,所以不会出现 冲突。

  1. //我们打开一个新的连接来模拟另外一个用户
  2. NorthwindDataContext otherUser_db = new NorthwindDataContext();
  3. var otherUser_product =
  4.   otherUser_db.Products.First(p => p.ProductID == 1);
  5. otherUser_product.UnitPrice = 999.99M;
  6. otherUser_db.SubmitChanges();
  7. //我们当前连接
  8. var product = db.Products.First(p => p.ProductID == 1);
  9. product.UnitPrice = 777.77M;
  10. try
  11. {
  12.   db.SubmitChanges();//当前连接执行成 功
  13. }
  14. catch (ChangeConflictException)
  15. {
  16. }
复制代码

说明:我们读取数据之后,另外一个用户获取并提交更新了这个数据,这时,我 们更新这个数据时,引起了一个并发冲突。系统发生回滚,允许你可以从数据库 检索新更新的数据,并决定如何继续进行您自己的更新。

  1. //当前 用户
  2. var product = db.Products.First(p => p.ProductID == 1);
  3. //我们打开一个新的连接来模拟另外一个用户
  4. NorthwindDataContext otherUser_db = new NorthwindDataContext() ;
  5. var otherUser_product =
  6.   otherUser_db.Products.First(p => p.ProductID == 1);
  7. otherUser_product.UnitPrice = 999.99M;
  8. otherUser_db.SubmitChanges();
  9. //当前用户修改
  10. product.UnitPrice = 777.77M;
  11. try
  12. {
  13.    db.SubmitChanges();
  14. }
  15. catch (ChangeConflictException)
  16. {
  17.   //发生异常!
  18. }
复制代码

Transactions事务

LINQ to SQL 支持三种事务模型,分别是:

显式本地事务:调用 SubmitChanges 时,如果 Transaction 属性设置为事务,则在同一事务的上下 文中执行 SubmitChanges 调用。成功执行事务后,要由您来提交或回滚事务。 与事务对应的连接必须与用于构造 DataContext 的连接匹配。如果使用其他连 接,则会引发异常。

显式可分发事务:可以在当前 Transaction 的作用 域中调用 LINQ to SQL API(包括但不限于 SubmitChanges)。LINQ to SQL 检 测到调用是在事务的作用域内,因而不会创建新的事务。在这种情况下, <token>vbtecdlinq</token> 还会避免关闭连接。您可以在此类事 务的上下文中执行查询和 SubmitChanges 操作。

隐式事务:当您调用 SubmitChanges 时,LINQ to SQL 会检查此调用是否在 Transaction 的作用域 内或者 Transaction 属性是否设置为由用户启动的本地事务。如果这两个事务 它均未找到,则 LINQ to SQL 启动本地事务,并使用此事务执行所生成的 SQL 命令。当所有 SQL 命令均已成功执行完毕时,LINQ to SQL 提交本地事务并返 回。

1.Implicit(隐式)

说明:这个例子在执行SubmitChanges ()操作时,隐式地使用了事务。因为在更新2种产品的库存数量时,第二个产品 库存数量为负数了,违反了服务器上的 CHECK 约束。这导致了更新产品全部失败 了,系统回滚到这个操作的初始状态。

  1. try
  2. {
  3.   Product prod1 = db.Products.First(p => p.ProductID == 4);
  4.   Product prod2 = db.Products.First(p => p.ProductID == 5);
  5.    prod1.UnitsInStock -= 3;
  6.   prod2.UnitsInStock -= 5;//错误:库存 数量的单位不能是负数
  7.   //要么全部成功要么全部失败
  8.    db.SubmitChanges();
  9. }
  10. catch (System.Data.SqlClient.SqlException e)
  11. {
  12.   //执行异常处理 
  13. }
复制代码

2.Explicit(显式)

说明:这个例子使用显式事 务。通过在事务中加入对数据的读取以防止出现开放式并发异常,显式事务可以 提供更多的保护。如同上一个查询中,更新 prod2 的 UnitsInStock 字段将使 该字段为负值,而这违反了数据库中的 CHECK 约束。这导致更新这两个产品的 事务失败,此时将回滚所有更改。

  1. using (TransactionScope ts = new TransactionScope())
  2. {
  3.   try
  4.   {
  5.      Product prod1 = db.Products.First(p => p.ProductID == 4);
  6.      Product prod2 = db.Products.First(p => p.ProductID == 5);
  7.     prod1.UnitsInStock -= 3;
  8.     prod2.UnitsInStock -= 5;//错误:库存数量的单位不能是负数
  9.     db.SubmitChanges();
  10.   }
  11.   catch (System.Data.SqlClient.SqlException e)
  12.    {
  13.     //执行异常处理
  14.   }
  15. }
复制代码

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值