11-4.在”模型定义”函数里调用另一个”模型定义”函数

问题

想要用一个”模型定义”函数去实现另一个”模型定义”函数

解决方案

假设我们已有一个公司合伙人关系连同它们的结构模型,如Figure 11-4所示:

Figure 11-4. A model representing the associate types in a company together with the reporting association

在我们的虚拟的公司里, , team members被一个team leader管理. Team leaders 被 project Managers管理. 而Supervisors管理 project managers.

如果我们想从给定的一个project manager 或supervisor返回所有他们管理的team members ,我们将需要通过  project managers 和team leaders才能获取到 team members. 为了隐藏导航透过这些层的复杂性,我们可以创建”模型定义”函数,来更简单地和直接地访问这些导航属性:

1. 在解决方案资源管理中右击.edmx 文件,打开方式 ➤ XML 编辑器.

2.把Listing 11-7 里的代码插入到概念层的<Schema> 标签下,这样就定义好了函数

Listing 11-7. Model-Defined Functions for Navigating the Associate Hierarchy

<Function Name="GetProjectManager" ReturnType="EFRecipesModel1104.ProjectManager">

<Parameter Name="teammember" Type="EFRecipesModel1104.TeamMember" />

<DefiningExpression>

treat(teammember.Manager.Manager as EFRecipesModel1104.ProjectManager)

</DefiningExpression>

</Function>

<Function Name="GetSupervisor" ReturnType="EFRecipesModel1104.Supervisor">

<Parameter Name="teammember" Type="EFRecipesModel1104.TeamMember" />

<DefiningExpression>

treat(EFRecipesModel1104.GetProjectManager(teammember).Manager as

EFRecipesModel1104.Supervisor)

</DefiningExpression>

</Function>

3.插入和查询模型的代码,如 Listing 11-8所示:

Listing 11-8. Using Both eSQL and LINQ to Query the Model

class Program

{

static void Main(string[] args)

{

RunExample();

Console.WriteLine("\nPress any key to exit...");

Console.ReadKey();

}

static void RunExample()

{

using (var context = new EFRecipesEntities1104())

{

context.Database.ExecuteSqlCommand("delete from chapter11.associate");

var john = new Supervisor { Name = "John Smith" };

var steve = new Supervisor { Name = "Steve Johnson" };

var jill = new ProjectManager

{

Name = "Jill Masterson",

Manager = john

};

var karen = new ProjectManager

{

Name = "Karen Carns",

Manager = steve

};

var bob = new TeamLead { Name = "Bob Richardson", Manager = karen };

var tom = new TeamLead { Name = "Tom Landers", Manager = jill };

var nancy = new TeamMember { Name = "Nancy Jones", Manager = tom };

var stacy = new TeamMember

{

Name = "Stacy Rutgers",

Manager = bob

};

context.Associates.Add(john);

context.Associates.Add(steve);

context.SaveChanges();

}

using (var context = new EFRecipesEntities1104())

{

Console.WriteLine("Using eSQL...");

var emps = context.Associates.OfType<TeamMember>()

//译注:这个where是下载的源码用的形式,但它怎么看也不是eSQL

.Where(q => q.Manager.Name == "Jill Masterson" || q.Manager.Name == "Steve Johnson");

//译注:这个where里书里的代码,倒像eSQL,可是找不到Where这个形式的重载,可能在某个扩展包里,有知晓,望赐教!

//                .Where(@"EFRecipesModel1104.GetProjectManager(it).Name = @projectManager ||

//EFRecipesModel.GetSupervisor(it).Name == @supervisor",

//                new ObjectParameter("projectManager", "Jill Masterson"),

//                new ObjectParameter("supervisor", "Steve Johnson"));

Console.WriteLine("Team members that report up to either");

Console.WriteLine("Project Manager Jill Masterson ");

Console.WriteLine("or Supervisor Steve Johnson");

foreach (var emp in emps)

{

Console.WriteLine("\tAssociate: {0}", emp.Name);

}

}

using (var context = new EFRecipesEntities1104())

{

Console.WriteLine();

Console.WriteLine("Using LINQ...");

var emps = from e in context.Associates.OfType<TeamMember>()

where MyFunctions.GetProjectManager(e).Name ==

"Jill Masterson" ||

MyFunctions.GetSupervisor(e).Name == "Steve Johnson"

select e;

Console.WriteLine("Team members that report up to either");

Console.WriteLine("Project Manager Jill Masterson ");

Console.WriteLine("or Supervisor Steve Johnson");

foreach (var emp in emps)

{

Console.WriteLine("\tAssociate: {0}", emp.Name);

}

}

}

}

public class MyFunctions

{

[EdmFunction("EFRecipesModel1104", "GetProjectManager")]

public static ProjectManager GetProjectManager(TeamMember member)

{

throw new NotSupportedException("Direct calls not supported.");

}

[EdmFunction("EFRecipesModel1104", "GetSupervisor")]

public static Supervisor GetSupervisor(TeamMember member)

{

throw new NotSupportedException("Direct calls not supported.");

}

}

上述Listing 11-8 代码输出如下:

Using eSQL...

Team members that report up to either

Project Manager Jill Masterson

or Supervisor Steve Johnson

Associate: Nancy Jones

Associate: Stacy Rutgers

Using LINQ...

Team members that report up to either

Project Manager Jill Masterson

or Supervisor Steve Johnson

Associate: Nancy Jones

Associate: Stacy Rutgers

它是如何工作的?

在如Listing 11-7所示的GetSupervisor() 函数中,我们需要做给 Manager 属性做三次导航.第一次从TeamMember导航到TeamLead, 第二次从TeamLead,导航到ProjectManager,最后一次从ProjectManager导航到Supervisor. 在Listing 11-7中,我们也定义了GetProjectManager() 函数,所以我们可以用这个函数来简化GetSupervisor() 函数.

我们使用eSQL的treat()运算符把一个公司合伙人实例转化为它们正确的类型 (ProjectManager 或Supervisor). 如果不使用这个运算符,EF将引发一个异常(不能遇到将Associate 映射为ProjectManager 或Supervisor)

在Listing 11-8中, 用 GetProjectManager() 和GetSupervisor() 函数,可以使我们的代码简洁些,因为隐藏了所有通过Manager导航属性穿越对象图的细节

因为我们函数不是返回一个IQueryable<T>, 所以我们不必为LINQ查询实现运行时方法,只需要用到它们的方法存根.

附:创建示例用到的数据库的脚本文件

Entity Framework 6 Recipes 2nd Edition(11-4)译 -> 在”模型定义”函数里调用另一个”模型定义”函数的更多相关文章

  1. Entity Framework 6 Recipes 2nd Edition 译 -> 目录 -持续更新

    因为看了<Entity Framework 6 Recipes 2nd Edition>这本书前面8章的翻译,感谢china_fucan. 从第九章开始,我是边看边译的,没有通读,加之英语 ...

  2. Entity Framework 6 Recipes 2nd Edition(9-3)译->找出Web API中发生了什么变化

    9-3. 找出Web API中发生了什么变化 问题 想通过基于REST的Web API服务对数据库进行插入,删除和修改对象图,而不必为每个实体类编写单独的更新方法. 此外, 用EF6的Code Fri ...

  3. Entity Framework 6 Recipes 2nd Edition(9-4)译->Web API 的客户端实现修改跟踪

    9-4. Web API 的客户端实现修改跟踪 问题 我们想通过客户端更新实体类,调用基于REST的Web API 服务实现把一个对象图的插入.删除和修改等数据库操作.此外, 我们想通过EF6的Cod ...

  4. Entity Framework 6 Recipes 2nd Edition(目录索引)

    Chapter01. Getting Started with Entity Framework / 实体框架入门 1-1. A Brief Tour of the Entity Framework ...

  5. Entity Framework 6 Recipes 2nd Edition(9-1)译->用Web Api更新单独分离的实体

    第九章 在N层结构的应用程序中使用EF 不是所有的应用都能完全地写入到一个单个的过程中(就是驻留在一个单一的物理层中),实际上,在当今不断发展的网络世界,大量的应用程序的结构包含经典的表现层,应用程, ...

  6. Entity Framework 6 Recipes 2nd Edition(11-1)译 -> 从“模型定义”函数返回一个标量值

    第11章函数 函数提供了一个有力代码复用机制, 并且让你的代码保持简洁和易懂. 它们同样也是EF运行时能利用的数据库层代码.函数有几类: Rowset Functions, 聚合函数, Ranking ...

  7. Entity Framework 6 Recipes 2nd Edition(13-2)译 -> 用实体键获取一个单独的实体

    问题 不管你用DBFirst,ModelFirst或是CodeFirst的方式,你想用实体键获取一个单独的实体.在本例中,我们用CodeFirst的方式. 解决方案 假设你有一个模型表示一个Paint ...

  8. Entity Framework 6 Recipes 2nd Edition(13-3)译 -> 为一个只读的访问获取实体

    问题 你想有效地获取只是用来显示不会更新的操作的实体.另外,你想用CodeFirst的方式来实现 解决方案 一个非常常见行为,尤其是网站,就是只是让用户浏览数据.大多数情况下,用户不会更新数据.在这种 ...

  9. Entity Framework 6 Recipes 2nd Edition(13-4)译 -> 有效地创建一个搜索查询

    问题 你想用LINQ写一个搜索查询,能被转换成更有效率的SQL.另外,你想用EF的CodeFirst方式实现. 解决方案 假设你有如下Figure 13-6所示的模型 Figure 13-6. A s ...

  10. Entity Framework 6 Recipes 2nd Edition(13-5)译 -> 使POCO的修改追踪更高

    问题 你正在使用POCO,你想提高修改跟踪的性能,同时使内存消耗更少.另外,你想通过EF的CodeFirst方式来实现. 解决方案 假设你有一个关于Account(帐户)和相关的Payments(支付 ...

随机推荐

  1. Windows平台分布式架构实践 - 负载均衡

    概述 最近.NET的世界开始闹腾了,微软官方终于加入到了对.NET跨平台的支持,并且在不久的将来,我们在VS里面写的代码可能就可以通过Mono直接在Linux和Mac上运行.那么大家(开发者和企业)为 ...

  2. submit text3常用快捷键

    在网上找了一些submit text的快捷键: Ctrl+D 选词 (反复按快捷键,即可继续向下同时选中下一个相同的文本进行同时编辑)Ctrl+G 跳转到相应的行Ctrl+J 合并行(已选择需要合并的 ...

  3. 一个诡异的COOKIE问题

    今天下午,发现本地的测试环境突然跑不动了,thinkphp直接跑到异常页面,按照正常的排错思路,直接看thinkphp的log 有一条 [ error ] [2]setcookie() expects ...

  4. Python(九) Python 操作 MySQL 之 pysql 与 SQLAchemy

    本文针对 Python 操作 MySQL 主要使用的两种方式讲解: 原生模块 pymsql ORM框架 SQLAchemy 本章内容: pymsql 执行 sql 增\删\改\查 语句 pymsql ...

  5. 脑洞大开之采用HTML5+SignalR2.0(.Net)实现原生Web视频

    目录 对SignalR不了解的人可以直接移步下面的目录 SignalR系列目录 前言 - -,我又来了,今天废话不多说,我们直接来实现Web视频聊天. 采用的技术如下: HTML5 WebRTC Si ...

  6. “RazorEngine.Templating.TemplateParsingException”类型的异常在 RazorEngine.NET4.0.dll 中发生,但未在用户代码中进行处理 其他信息: Expected model identifier.

    这个问题是由于在cshtml中 引用了model这个单词  它可能和Model在解析时有冲突. 解决方法:把model换成别的单词就可以了.

  7. Windos环境用Nginx配置反向代理和负载均衡

    Windos环境用Nginx配置反向代理和负载均衡 引言:在前后端分离架构下,难免会遇到跨域问题.目前的解决方案大致有JSONP,反向代理,CORS这三种方式.JSONP兼容性良好,最大的缺点是只支持 ...

  8. 基于开源项目SharpMap的热力图(HeatLayer)实现。

    当前公司需要一个用时较少的热力图呈现方案,在避免较底层的GDI开发和比较了多家GIS产品的实际效果之后,团队决定用sharpMap的API来实现,由于之前框架采用的是另外一个开源项目GMap.net, ...

  9. 解决:SharePoint当中的STP网站列表模板没有办法导出到其它语言环境中使用

    首在在你的英文版本上,导出列表或是网站的模板,这个文件可能是这样滴:template.stp 把这个文件 template.stp 命名为 template.cab 解压 这个 *.cab 文件 在解 ...

  10. Missing Push Notification Entitlement 问题

    最近打包上传是遇到一个问题: 描述: Missing Push Notification Entitlement - Your app includes an API for Apple's Push ...