在前一篇文章中,我提出了在使用LINQ to SQL进行更新操作时可能会遇到的几种问题。其实这并不是我一个人遇到的问题,当我在互联网上寻找答案时,我发现很多人都对这个话题发表过类似文章。但另我无法满足的是,他们尽管提出了问题,却没有进行详细的剖析,只给出了解决方案(如添加RowVersion列、去除关联等),但却没有说明为什么必须这么做。这也是我写上篇的初衷,希望通过对LINQ to SQL源代码的分析,来一步一步找出解决问题的办法。本文将对这些方法一一进行讨论。

方案一:重新赋值

在TerryLee、Anytao和Ding Xue等人的开源框架Ezsocio中,有些地方采取了重新赋值的方法。在Update方法内部,根据主键获取数据库中的实体,然后与参数中的实体对其属性一一赋值。

public void UpdateProfile(Profile p)
{using (RepositoryContext db = new RepositoryContext()){var profile = db.GetTable<Profile>().First<Profile>(u => u.ID == p.ID);profile.Birthday = p.Birthday;profile.Gender = p.Gender;profile.Hometown = p.Hometown;profile.MSN = p.MSN;profile.NickName = p.NickName;profile.PhoneNumber = p.PhoneNumber;profile.QQ = p.QQ;profile.State = p.State;profile.TrueName = p.TrueName;profile.StateRefreshTime = p.StateRefreshTime;profile.Avatar = p.Avatar;profile.Website = p.Website;db.SubmitChanges();}
}

杨过兄也同样给出了该方案的反射方法,实现属性值的自动拷贝。

但我个人认为这是一种避实就虚的方案,没有使用LINQ to SQL提供的用于更新操作的API,而采取了一种迂回的策略。这其实是一种妥协,难道因为Attach方法“不好用”,我们就不用了吗?呵呵。

方案二:禁用对象跟踪

对此,lea提出可以通过将DataContext的ObjectTrackingEnabled属性设置为false,来达到正确更新的目的。

public Product GetProduct(int id)
{NorthwindDataContext db = new NorthwindDataContext();db.ObjectTrackingEnabled = false;return db.Products.SingleOrDefault(p => p.ProductID == id);
}

其他的代码没有任何变化。

为什么禁用对象跟踪之后,就能正常更新了呢?我们还是从源代码中来寻找答案吧。

public bool ObjectTrackingEnabled
{get{this.CheckDispose();return this.objectTrackingEnabled;}set{this.CheckDispose();if (this.Services.HasCachedObjects){throw System.Data.Linq.Error.OptionsCannotBeModifiedAfterQuery();}this.objectTrackingEnabled = value;if (!this.objectTrackingEnabled){this.deferredLoadingEnabled = false;}this.services.ResetServices();}
}

原来设置ObjectTrackingEnabled为false时,会同时将DeferredLoadingEnabled设置为false。这样,在执行查询时,将不会为实体加载任何需延迟查询的数据,因此Attach时也不会抛出异常(见上篇的分析)。

在MSDN中我们还得到下面这条有用的信息:将ObjectTrackingEnable属性设置为false,可以提高检索时的性能,因为这样可以减少要跟踪的项目。这真是一个很有诱惑的特性。

但禁用对象跟踪时,要特别注意两点:(1)必须在执行查询前禁用。(2)禁用之后不能再调用Attach和SubmitChanges方法。否则都将引发异常。

方案三:移除关联

在前一篇文章中已经介绍一个蹩脚的方法,即在GetProduct方法中手动设置与Product关联的Category为null。我们可以把这部分代码提取出来,放入一个Detach方法中。因为这个Detach是实体的方法,可以使用分部类:

public partial class Product
{public void Detach(){this._Category = default(EntityRef<Category>);}
}
public partial class Category
{public void Detach(){foreach (var product in this.Products){product.Detach();}}
}

但是这种对每个实体都定义Detach的方法过于繁琐。随着实体的增多,关系越来越复杂,很容易出现漏掉的属性。张逸提出了一个非常优雅的方法,利用反射对该逻辑进行抽象:

private void Detach(TEntity entity)
{foreach (FieldInfo fi in entity.GetType().GetFields(BindingFlags.NonPublic | BindingFlags.Instance)){if (fi.FieldType.ToString().Contains("EntityRef")){var value = fi.GetValue(entity);if (value != null){fi.SetValue(entity, null);}}if (fi.FieldType.ToString().Contains("EntitySet")){var value = fi.GetValue(entity);if (value != null){MethodInfo mi = value.GetType().GetMethod("Clear");if (mi != null){mi.Invoke(value, null);}fi.SetValue(entity, value);}}}
}

也有人认为在Detach时应该把PropertyChanging和PropertyChanged事件设置为null,但总体的思路是一样的。

方案四:使用委托

这是ZC29同学在我上一篇文章的评论里给出的方法,我个人认为非常值得借鉴。

public void UpdateProductWithDelegate(Expression<Func<Product, bool>> predicate, Action<Product> action)
{NorthwindDataContext db = new NorthwindDataContext();var product = db.Products.SingleOrDefault(predicate);action(product);db.SubmitChanges();
}
// Client code
ProductRepository repository = new ProductRepository();
repository.UpdateProductWithDelegate(p => p.ProductID == 1, p =>{p.ProductName = "Changed";});

使用Lambda表达式将GetProduct的逻辑植入UpdateProduct中,并且使用委托将更新逻辑也延缓执行,这样巧妙地将查找和更新放进了一个DataContext里,从而绕开了Attach。但是这种方法API有些过于复杂,对客户端编程人员的水平要求过高。而且在Update里还要执行一遍Get的逻辑,尽管性能上的损失微乎其微,但看上去总多多少少给人一种不够DRY的感觉。

方案五:使用UPDATE语句

在Ezsocio的源代码中,我发现了RepositoryBase.UpdateEntity方法。在方法内部进行SQL语句的拼接,并且将只更新发生更改的列。由于此处已经不再使用ITable,并且需要完整的框架支持,因此不再进行过多的评述。详情请参考Ezsocio的源代码。

总结

本文列举了近几天我在互联网上找到的几种解决方案,它们各有利弊,孰优孰劣,见仁见智。在下篇中,我将对这几种方法进行性能上的比较,从而找出最优方案。

相关文章

  • 使用LINQ to SQL更新数据库(上):问题重重
  • 使用LINQ to SQL更新数据库(中):几种解决方案

转载于:https://www.cnblogs.com/waw/archive/2011/08/29/2156985.html

艾伟_转载:使用LINQ to SQL更新数据库(中):几种解决方案相关推荐

  1. 艾伟_转载:使用LINQ to SQL更新数据库(上):问题重重

    在学习LINQ时,我几乎被一个困难所击倒,这就是你从标题中看到的更新数据库的操作.下面我就一步步带你走入这泥潭,请准备好砖头和口水,Follow me. 从最简单的情况入手 我们以Northwind数 ...

  2. 转载:LINQ to SQL更新数据库操作

    翻译整理ScottGu的关于LINQ to SQL的Part 4: Updating our Database .该Post讲解了如何使用LINQ to SQL更新数据库,以及如何整合业务逻辑和自定义 ...

  3. 艾伟_转载:简单的自动更新程序实现

    本文将演示一种桌面程序自动更新方案,其步骤比较多,但原理非常简单,通用性尚可,对于小型应用来说,直接拿去就可以用了. 原理 服务器端的结构是这样的: 其工作原理如下: Update.asmx仅提供一个 ...

  4. mvc mysql linq_MVC3+Linq to sql 显示数据库中数据表的数据

    1:首先创建asp.net mvc3应用程序 2:创建项目完成后 找到controllers文件鼠标右击选择添加控制器 3 为models文件夹添加一个linq to sql类文件,然后把数据库中的数 ...

  5. 数据库LINQ TO SQL在Silverlight中的应用(WCF)------学习笔记(一)

    数据库LINQ TO SQL在Silverlight中的应用(WCF)------学习笔记(一) 步骤: 1. 创建SILVERLIGHT应用程序 2. 创建LINQ TO SQL [注意序列化的问题 ...

  6. Linq to Sql: 集成数据库语言查询之一

    Linq to Sql: 集成数据库语言查询之一 2007-09-11 11:30:28 来源:天极yesky 作者:随风流月 带您探索"CRUD "操作-创建,接收,更新与删除, ...

  7. SQL更新表中数据语句

    如果,你想更新数据库中其中某一条数据.那么你需要使用UPDATE语句.基本的UPDATE语句如下. UPDATE <表名> SET 字段1=值1, 字段2=值2, ... WHERE .. ...

  8. 通过VB向SQL Server数据库中录入数据

    Private Declare Sub Sleep Lib "kernel32" (ByVal dwMilliseconds As Long) 一.数据录入 通过VB向SQL Se ...

  9. 浅析SQL Server数据库中的伪列以及伪列的含义

    浅析SQL Server数据库中的伪列以及伪列的含义 原文:浅析SQL Server数据库中的伪列以及伪列的含义 本文出处:http://www.cnblogs.com/wy123/p/6833116 ...

最新文章

  1. 为什么我们有时不用配置java环境变量?
  2. [OS复习]操作系统综述2
  3. OSPF动态路由协议(了解ospf,RIP协议,外部网关协议内部网关协议,ospf工作过程,选举过程,ospf邻居关系7个状态)
  4. 中国的程序员为什么这么辛苦?
  5. 你是外包,麻烦不要偷吃零食,注意素质...
  6. java.util.ResourceBundle用法
  7. jquery中的html代码、文本以及值
  8. 测试的第二重境界:站在Bug之上
  9. 【2018CPCP-Final G:】Pastoral Life in Stardew Valley
  10. C语言教材市场的分析
  11. python 人脸识别活体检测_手把手教你用Python实现人脸识别,辨别真假!
  12. 大数据对于企业的价值,主要体现在哪几个方面?
  13. 孙子兵法始计篇读后感&心得(下)
  14. 微信公众号文章音视频下载的几种办法-涵盖PC端和手机端
  15. 1.3寸 SH1106 OLED 屏幕驱动 基于stm32
  16. EXCEL地理工具--小O地图EXCEL插件0705版 2022.4.28发布
  17. 美国数据科学家带你看看大数据的未来
  18. 在线交易系统 服务器1,金字塔决策交易系统金钻版服务器及客户端安装配置说明1.doc...
  19. 河南省高中计算机会考难不难,高中会考难不难
  20. import as在python_python import as教程

热门文章

  1. java 优化线程_Java | 多线程调优(下):如何优化多线程上下文切换?
  2. 君君喂大象C语言答案,2017年北师大版二年级语文上册句子专项复习题及答案
  3. 鸿蒙os芯片,没有了芯片,华为能靠鸿蒙OS系统打出差异化吗?
  4. telegr怎么连接不上_无线网密码正确但是手机连接不上wifi?
  5. linux右上角不显示网络连接_来体验下Linux吧
  6. 鸿蒙系统下载 绿色,PGWARE PcMedik
  7. mysql inputoutput_PHP:同时使用INPUT和OUTPUT参数(不“ INOUT”)调用MySQL存储过程
  8. bmp转换tiff c++代码_如何用Java语言将图像转换为PDF?Spire.PDF for Java轻松搞定
  9. python一般用什么软件写_python用什么软件写代码
  10. 作者:姜春宇(1987-),男,中国信息通信研究院移动互联网与大数据部工程师,数据中心联盟大数据技术与产品工作组组长。...