提示13. 附加一个实体的简单方式

提示13. 附加一个实体的简单方式

问题:

在早先的一些提示中,我们讨论了使用Attach来加载一个处于未改变(unchanged)状态的东西到ObjectContext从而避免进行查询的开销。

如果性能是你的目标,Attach就是要选择的武器。

不幸的是我们的API不能适应99%的情况,即每个类型仅有一个实体集(entity set)的情况。Entity Framework支持单类型多实体集(Multiple Entity Sets perType)或称MEST,且API反映了这一点,要求你提供你要附加的实体集(EntitySet)的名称。

即,像这样:

1 ctx.Attach("Orders", order); 

如果你像我一样,可能你也会发反感在代码里硬编码入字符串。它们容易出错且这类东西会污染你的代码,这实质上是一个"小问题"。

.NET 4.0中的解决方案

在.NET 4.0中通过每一个EntitySet返回ObjectSet<T>而不是ObjectQuery<T>这个强类型的属性修复了这个问题。ObjectSet<T>有Add, Delete及Attach方法直接处理这个问题,所以你可以写如下这样的代码:

1 ctx.Order.Attach(order);

没有一个字符串出现!

这种解决方案是理想的,你附加需要的实体集,且无论你是否有MEST,它都工作。

.NET 3.5中的解决方案

.NET 3.5中应该怎么办呢?

我的观点是,我们应该提供一个泛型版本的Attach,即如下这样:

1 void AttachToDefaultSet<T>(T Entity); 

这个方法会检查T中存在多少个EntitySet,如果只有一个,其将附加这个实体到那个实体集。然而如果多于一个其将抛出异常。

虽然这种方法没有扩展方法的力量,但它也很容易写。

以下是你需要做的:

  1. 得到<T>类型的EntityType
  2. 得到这个EntityType可能属于的EntitySet类型的列表。EntityType可能为派生类型(像Car),并且实际上属于一个父类型集合(像Vehicles)。
  3. 遍历EntityContainer,对每一个EntityContainer的EntitySet查找一个匹配。
  4. 如果找到一个则进行附加,否则抛出异常。

下面让我们来完成:

但首先要注意,这只是示例级质量的代码,我是一个项目经理而非程序员,所以使用这些代码的风险自担:)

首先我们添加一个扩展方法到MetadataWorkspace来获取一个CLR类型(O-Space)对应的概念模型(C-Space)EntityType。

 1 public static EntityType GetCSpaceEntityType<T>(
 2        this MetadataWorkspace workspace
 3 )
 4 {
 5     if (workspace == null)
 6         throw new ArgumentNullException("workspace");
 7     // Make sure the assembly for "T" is loaded
 8     workspace.LoadFromAssembly(typeof(T).Assembly);
 9     // Try to get the ospace type and if that is found
10     // look for the cspace type too.
11     EntityType ospaceEntityType = null;
12     StructuralType cspaceEntityType = null;
13     if (workspace.TryGetItem<EntityType>(
14         typeof(T).FullName,
15         DataSpace.OSpace,
16         out ospaceEntityType))
17     {
18         if (workspace.TryGetEdmSpaceType(
19             ospaceEntityType,
20             out cspaceEntityType))
21         {
22             return cspaceEntityType as EntityType;
23         }
24     }
25     return null;
26 }

由于你可能在<T>的元数据加载前调用这段代码,代码第一行保证了<T>这个程序集被加载,如果程序集已被加载其不会执行任何操作。

下一步我们添加一个方法得到一个我们需要匹配的所有类型的枚举,即,包括当前类型在内的父类型的层级:

 1 public static IEnumerable<EntityType> GetHierarchy(
 2     this EntityType entityType)
 3 {
 4     if (entityType == null)
 5         throw new ArgumentNullException("entityType");
 6     while (entityType != null)
 7     {
 8         yield return entityType;
 9         entityType = entityType.BaseType as EntityType;
10     }
11 }

最后,我们可以开始完成AttachToDefaultSet方法:

 1 public static void AttachToDefaultSet<T>(
 2     this ObjectContext ctx,
 3     T entity)
 4 {
 5     if (ctx== null) throw new ArgumentNullException("ctx");
 6     if (entity == null) throw new ArgumentNullException("entity");
 7
 8
 9     MetadataWorkspace wkspace = ctx.MetadataWorkspace;
10     EntitySet set = wkspace
11        .GetEntitySets(wkspace.GetCSpaceEntityType<T>())
12        .Single();
13
14     ctx.AttachTo(set.Name, entity);
15 }

这里使用了标准的.Single()方法,如果不是恰好存在一个对应EntityType的可能的实体集其将抛出一个异常。

使用这个实现,我们可以将前文的代码使用下面这种方式重写:

1 Product p = newProduct { ID = 1, Name = "Chocolate Fish" } ctx.AttachToDefaultSet(p);

当然,除非你使用MEST...但你可能不使用!

附加说明

虽然这段代码可以很好的工作,但其确实没有进行过任何优化。

或许缓存对应一个CLR类型的可能的集合的名字是有意义的,这样当你进行Attach的时候就无需再进行相同的检查,这就当留给你的练习了!

提示索引

是的,这里有一个本系列剩余提示的索引

提示14. 怎样缓存Entity Framework引用数据

场景:

为了使应用程序可以工作,缓存常用的引用数据是非常有意义的。

引用数据的好例子包括像States, Countries, Departments等事物。

通常你想要这些数据随时可用,以方便的进行填充下拉列表等操作。

什么地方将引用数据缓存在手边的一个好例子是让新客户注册的页面,表单的一部分收集用户的地址,包括他们的州。在这个例子中你需要引用数据做两件事:

1. 构建表单中选择州的下拉列表。

2. 将州指定到最终的用户记录。

你怎样使用Entity Framework来支持这类场景呢?

解决方案:

当设计解决方案时我们需要记住两个关键点。

1. 一个实体同一时间只能被附加到一个ObjectContext,至少.NET 3.5 SP1中是这样。

2. 你可能由许多线程同时使用缓存的引用数据(读ObjectContexts)。

本质上这两点相互矛盾。

解决方案是当我们由缓存读取时拷贝实体,这样附加拷贝将不会影响任何其它线程。

如果这是一个webform解决方案,我们可能要写这样的代码:

 1 var customer = new Customer{
 2    Firstname = txtFirstname.Text,
 3    Surname = txtSurname.Text,
 4    Email = txtEmail.Text,
 5    Street = txtStreet.Text,
 6    City = txtCity.Text,
 7    State = statesCache.GetSingle(
 8       s => s.ID = Int32.Parse(ddState.SelectedValue)
 9    ),
10    Zip = txtZip.Text
11 }
12 ctx.AddToCustomers(customer);
13 ctx.SaveChanges();

但是这有一个大问题。当你添加customer到ObjectContext时,拷贝的State也是added状态。如果我们这样做,Entity Framework会认为其需要将State插入到数据库。而这不是我们想要的。

所以我们需要通过使用 AttachTo(...) 告诉Entity Framework State这个拷贝的已经存在于数据库中:

1 var state = statesCache.GetSingle(
2      s => s.ID = Int32.Parse(ddState.SelectedValue)
3 );
4 // See Tip 13 to avoid specifying the EntitySet
5 // as a string
6 ctx.AttachTo("States", s); 

然后我们可以继续构建customer:

 1 var customer = new Customer{
 2    Firstname = txtFirstname.Text,
 3    Surname = txtSurname.Text,
 4    Email = txtEmail.Text,
 5    Street = txtStreet.Text,
 6    City = txtCity.Text,
 7    State = state,
 8    Zip = txtZip.Text
 9 }
10 ctx.SaveChanges();

如果你足够警惕,你可能已经发现我没有再次调用 AddToCustomers(...) 。

为什么呢?嗯,当你构建一个到已存在于context (State = state)中的关系时,这个customer会自动被添加。

现在,在 SaveChanges() 被调用时,只有Customer被存储到数据库。State根本不会被持久化,因为Entity Framework认为其不曾改变。

有趣的是,我们可以利用State不被支持化的事实作为我们的条件。

因为,State的主键属性是Entity Framework构建关系时唯一需要知道的,即使其它属性都错了也没有关系,主键属性实际上是我们拷贝时唯一需要的。

这样,我们的拷贝代码可以非常简单:

1 public State Clone(State state)
2 {
3    return new State {ID = state.ID};
4 }

或者使用如下lambda表达式:

1 var cloner = (State s) => new State {ID = s.ID};

只要我们不想要修改拷贝,这些就是所有我们实际需要的。

现在我们知道了需要什么,编写一个非常简单的提供缓存与"只读拷贝"服务的泛型类就很容易了。

 1 public class CloningCache<T> where T : class
 2 {
 3     private List<T> _innerData;
 4     private Func<T, T> _cloner;
 5     public CloningCache(IEnumerable<T> source, Func<T, T> cloner)
 6     {
 7         _innerData = source.ToList();
 8         _cloner = cloner;
 9     }
10     public T GetSingle(Func<T, bool> predicate)
11     {
12         lock (_innerData)
13         {
14             return _innerData
15                         .Where(predicate)
16                         .Select(s => _cloner(s))
17                         .Single();
18         }
19     }
20 }

注意, GetSingle(...) 方法拷贝它找到的结果。

另外使用这个拷贝缓存非常简单:

1 var statesCache = new CloningCache<State>(
2       ctx.States,
3       (State s) => new State {ID = s.ID}
4 );

构造函数的第一个参数是要缓存的数据(即数据库中所有的States),第二个参数是怎样实现拷贝,我们需要跨多个ObjectContext安全的使用这个缓存。

一旦你初始化了这个缓存(大概是在Global.asax中),无论什么情况下你需要直接访问引用数据,你都可以在一个静态变量中情况此缓存。

如果哪里讲的不清楚或你有什么问题,请告诉我。

提示15. 怎样避免加载非必须的属性

更新:对之前需要Original值做了一系列重要的更正。

问题:

想象如果你查询博客随笔:

1 var myPosts = from post in ctx.Posts
2               orderby post.Created descending
3               select post;

仅仅这样你就可以输出博客标题等等。

1 foreach(var post in myPosts)
2 {
3      Console.WriteLine("{0} on {1}", post.Title, post.Created);
4 }

这样你做了一大些无用的工作来加载你实际上不需要的属性。

只读解决方案:

对于只读场景,解决方案很容易。

你只需进行投影操作:

1 var myPosts = from post in ctx.Posts
2               orderby post.Created descending
3               select new {post.Title, post.Created};

这样你就避免了加载你实际上不需要的属性。

对于有许多属性或存在映射到数据库中一个blob列的属性的实体,例如像Body之类映射到一个nvarchar(max)列的属性,这尤其重要。

读写解决方案:

但如果你需要修改实体该怎么办呢?

此处投影不是一个好方案,因为除非你获取一个完整的对象,否则你将不会得到任何对象服务,这意味着将无法更新。

嗯…

一如往常,得到一个解决方案的关键的就是理解Entity Framework的工作方式。

当更新一个实体,Entity Framework将更新以如下格式发送到数据库(伪代码):

 1 UPDATE [Table]
 2 SET
 3      ModifiedProperty1 = NewValue1,
 4      ModifiedProperty2 = NewValue2,
 5      ...
 6      ModifiedPropertyN = NewValueN
 7 WHERE
 8      KeyProperty = KeyValue AND
 9      ModifiedProperty1 = OriginalValue1 AND
10      ModifiedProperty2 = OriginalValue2 AND
11      ...
12      ModifiedPropertyN = OriginalValueN

注意没有修改过的属性不会出现在更新命令的任何地方。

重大发现:这意味着你只需要知道主键属性的原始值即可。*

带着这些发现,我们可以进行下面这样的尝试:

  1. 仅投射出我们需要读写的列
  2. 由投影伪造一个实体,忽略我们不关心的列。
  3. Attach(...)那个"部分正确"的实体
  4. 对实体做所需的更改
  5. SaveChanges(...)

这样我们就可以在不实例化我们不感兴趣的属性的情况下更改我们的实体。

下面是一些可以完成上述工作的代码:

 1 // Project just the columns we need
 2 var myPosts = from post in ctx.Posts
 3               orderby post.Created descending
 4               select new {post.ID, post.Title};
 5 // Fabricate new Entities in memory.
 6 // Notice the use of AsEnumerable() to separate the in db query
 7 // from the LINQ to Objects construction of Post entities.
 8 var fabricatedPosts = from p in myPosts.AsEnumerable()
 9     select new Post{ID = p.ID, Title = post.Title};
10 // Now we attach the posts
11 // And call a method to modify the Title
12 foreach(var p in fabricatedPosts)
13 {
14     ctx.AttachTo("Posts", p);
15     p.Title = ChangeTitle(p.Title);
16 }
17 ctx.SaveChanges();

注意我们只检索了ID属性(主键)和Title属性(我们要修改的东西),但我们仍成功地进行了更新。

TA DA!

*警告/并发问题

如果你使用存储过程更新实体,这个提示中的内容不适用。

如果你考虑存储过程工作的方式你可以明白为什么。当使用存储过程进行更新时,所有当前值(及部分原始值)被映射到参数,而不管它们是被修改过。这基本意味着你不得不获得所有原始值:(

另外有些时候你需要告诉Entity Framework一些其它的原始值,因为没有它们更新不会成功:

  • 并发属性:并发属性的原始值被包含在更新语句中,来保证你只可以在明确知道当前数据库版本的情况下更新数据库。所以没有正确的原始值更新不会成功。
  • EntityFramework的EntityKey值:你也需要知道0..1关系的原始值,即使不准备改变此关系。例如如果一个订单有一个客户,你将需要知道CustomerReference.EntityKey,然后你可以使用这个已确立的关系初始化一个新的Order。当你使用FK属性(在即将到来的.NET 4.0中)时,这个问题将不复存在。

C-Side映射条件引用的属性:C-Side映射条件的值用于算出应用了哪个映射,所以没有正确的原始值就不会确立正确的更新映射。大部分人不会使用这个特性。

时间: 2024-08-04 21:22:41

提示13. 附加一个实体的简单方式的相关文章

Entity Framework 6 Recipes 2nd Edition(10-1)译-&gt;非Code Frist方式返回一个实体集合

存储过程 存储过程一直存在于任何一种关系型数据库中,如微软的SQL Server.存储过程是包含在数据库中的一些代码,通常为数据执行一些操作,它能为数据密集型计算提高性能,也能执行一些为业务逻辑. 当你使用数据的时候,有时你会通过存储过程来获取它们. 在本章, 我们探讨一些EF在使用存储过程时,需要关注的地方.我们在本书的其它章节也使用了存储过程, 但通常都是context为执行插入.更新和删除动作. 在本章,我们将为你展示多种使用存储过程的方式. 10-1. 非Code Frist方式返回一个

foreach ()语法结构提供了遍历数组的简单方式

foreach 语法结构提供了遍历数组的简单方式.foreach 仅能够应用于数组和对象,如果尝试应用于其他数据类型的变量,或者未初始化的变量将发出错误信息.有两种语法: foreach (array_expression as $value) statement foreach (array_expression as $key => $value) statement 第一种格式遍历给定的 array_expression 数组.每次循环中,当前单元的值被赋给 $value 并且数组内部的指

在java 中一种简单方式的声明静态Map常量的方法

我现在需要在一个类里面放一个HashMap,往里面放一些数据,每次要从数据库中取数据的时候先查找HashMap,看是否已经存在,若存在就直接提取,若不存在就从数据库中抽取数据之后再放到HashMap中,那是否应该把HashMap设置为类的静态变量啊? 若这样设置public static HashMap hmData :就无法往hmSpace 中加入数据了,是否应该New一个对象呢? 常用的声明方式(使用静态代码块): Java代码 public final static Map map = n

T4 生成实体和简单的CRUD操作

<#@ template debug="false" hostspecific="false" language="C#" #> <#@ assembly name="System.Core.dll" #> <#@ assembly name="System.Data.dll" #> <#@ assembly name="System.Data.DataS

java 程序执行输出有两种简单方式

java 程序执行输出有两种简单方式: 1. System.out.println("需要输出的内容"): 该方法可参看运行一个简单的Java程序 结果图: 2. System.out.print("需要输出的内容"): 1 public class HelloWorld 2 { 3 //Java程序的入口方法,程序将从这里开始运行 4 public static void main(String[] args) 5 { 6 //向控制台打印一条语句 7 Syste

每天一个设计模式-1 简单工厂

每天一个设计模式-1  简单工厂 1.简单工厂的定义 提供一个创建对象实例的功能,而无须关心其具体实现(核心). 虽然不能让模块外部知道模块内部的具体实现,但模块内部是可以知道具体实现类的.干脆在模块内部建一个类,用这个类来创建接口,然后把创建号的接口返回给客户端:这样,外部应用就只需要根据这个类来获取相应的接口对象,通过这个接口对象就可以操作接口定义的方法了.显然,这个类就像一个工厂,专门用来生成(生产)需要的接口对象. 2.简单的例子 说明: 代码: Api:接口,wear是一个公有方法.

练习13.13的一个有意思的现象

在做C++ Primer 5th的联系13.13时有一个很有意思的现象:当一个声明一个保存自定义类类型的vector时,如果在进行push_back操作之前这个vector的capacity和其size一致的话,则程序会向操作系统申请更多的内存以保存更多的元素.此时,整个vector会对其已经构建的元素重新全部构建一次(即调用拷贝构造函数),然后再在末尾元素之后构建新的元素.最后调用析构函数销毁扩容之前构建的元素. 下面直接看程序: //X结构体的定义 struct X { //默认构造函数 X

canvas-在画布中画两个方块(一个空心一个实体)

效果图: 代码: 1 <canvas id="c1" width="400" height="400" style="background-color:red"> 2 </canvas> 3 4 <script type="text/javascript"> 5 //获取画布元素 6 var canvas=document.getElementById("c1&

javascript 跟随鼠标移动的提示框的一个小demo

下面提供一种跟随鼠标移动的提示框的思路,方便在以后工作中应用,主要是应用到鼠标移动产生的数值来进行移动提示框的定位... CSS代码   .box{height:100px;width:100px;background:orange;position:relative;margin:40px;} .move{height:20px;width:20px;background:red;position:absolute;left:0px;top:0px;display:none;} HTML代码