11-6.从一个”模型定义”函数里返回一个复杂类型

问题

想要从一个”模型定义”函数返回一个复杂类型

解决方案

假设我们有一个病人(patient)和他们访客(visit)的模型,如 Figure 11-6所示 .

Figure 11-6. A model for patient visits

我们想要创建一个”模型定义”函数,返回一个概要信息,包括:病人名字,病人的访客数,和病人累积的账单. 此外,我们只过滤出年龄超过40岁的病人:

1. 在模型设计视图上,右击, 新建 ➤ 复杂类型.

2.在模型浏览器里右击新建的复杂类型,重命名为VisitSummary, 然后给复杂属性添加下列属性::

a. Name: String,不可为null

b. TotalVisits: Int32, 不可为null

c. TotalCost:Decimal, 不可为null

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

4. 在.edmx 文件的概念模型conceptual models)的<Schema> 标签下插入Listing 11-11所示的代码,这样函数主定义好了.

 Listing 11-11. The GetVisitSummary() Model-Defined Function

<Function Name="GetVisitSummary"  ReturnType="Collection(EFRecipesModel.VisitSummary)">

<DefiningExpression>

select VALUE EFRecipesModel.VisitSummary(pv.Patient.Name,

Count(pv.VisitId),Sum(pv.Cost))

from EFRecipesEntities.PatientVisits as pv

group by pv.Patient.Name

</DefiningExpression>

</Function>

5. 用如 Listing 11-12.所示的代码来插入和查询这个模型:

Listing 11-12. Using eSQL and LINQ with the VisitSummary() Function to Query the Model

class Program

{

static void Main(string[] args)

{

RunExample();

}

static void RunExample()

{

using (var context = new EFRecipesEntities())

{

string hospital = "Oakland General";

var p1 = new Patient { Name = "Robin Rosen", Age = 41 };

var p2 = new Patient { Name = "Alex Jones", Age = 39 };

var p3 = new Patient { Name = "Susan Kirby", Age = 54 };

var v1 = new PatientVisit

{

Cost = 98.38M,

Hospital = hospital,

Patient = p1

};

var v2 = new PatientVisit

{

Cost = 1122.98M,

Hospital = hospital,

Patient = p1

};

var v3 = new PatientVisit

{

Cost = 2292.72M,

Hospital = hospital,

Patient = p2

};

var v4 = new PatientVisit

{

Cost = 1145.73M,

Hospital = hospital,

Patient = p3

};

var v5 = new PatientVisit

{

Cost = 2891.07M,

Hospital = hospital,

Patient = p3

};

context.Patients.Add(p1);

context.Patients.Add(p2);

context.Patients.Add(p3);

context.SaveChanges();

}

using (var context = new EFRecipesEntities())

{

Console.WriteLine("Query using eSQL...");

var esql = @"Select value ps from EFRecipesEntities.Patients

as p join EFRecipesModel.GetVisitSummary()

as ps on p.Name = ps.Name where p.Age > 40";

var objectContext = (context as IObjectContextAdapter).ObjectContext;

var patients = objectContext.CreateQuery<VisitSummary>(esql);

foreach (var patient in patients)

{

Console.WriteLine("{0}, Visits: {1}, Total Bill: {2}",

patient.Name, patient.TotalVisits.ToString(),

patient.TotalCost.ToString("C"));

}

}

using (var context = new EFRecipesEntities())

{

Console.WriteLine();

Console.WriteLine("Query using LINQ...");

//译注:遇到了与11-5一样的异常

var patients = from p in context.Patients

join ps in context.GetVisitSummary() on p.Name equals

ps.Name

where p.Age >= 40

select ps;

foreach (var patient in patients)

{

Console.WriteLine("{0}, Visits: {1}, Total Bill: {2}",

patient.Name, patient.TotalVisits.ToString(),

patient.TotalCost.ToString("C"));

}

}

}

}

partial class EFRecipesEntities

{

[EdmFunction("EFRecipesModel", "GetVisitSummary")]

public IQueryable<VisitSummary> GetVisitSummary()

{

var objectContext = (this as IObjectContextAdapter).ObjectContext;

return objectContext.CreateQuery<VisitSummary>(

Expression.Call(Expression.Constant(this),

(MethodInfo)MethodInfo.GetCurrentMethod()).ToString());

}

}

Listing 11-12代码输出结果如下:

Query using eSQL...

Robin Rosen, Visits: 2, Total Bill: $1,221.36

Susan Kirby, Visits: 2, Total Bill: $4,036.80

Query using LINQ...

Robin Rosen, Visits: 2, Total Bill: $1,221.36

Susan Kirby, Visits: 2, Total Bill: $4,036.80

它是如何工作的?

我们先在模型里添加一个复杂类型,接着创建如Listing 11-11的GetVisitSummary() 函数,它能返回包含这个新建的复杂类型的集合.注意:复杂类型的构造函数接受参数的顺序要与我们定义它的属性时的顺序一致. 你可能需要检查.edmx文件,确保设计器按我们添加的顺序是一致.

由于我们函数返回IQueryable<VisitSummary>, 所以我们需要实现引导代码. 同样,因为我们需要访问ObjectContext的QueryProvider,所以我们需要在EFRecipesEntities类中实现运行时方法.

如果我们把这个函数用在LINQ查询中,你可能需要让这个方法返回IQueryable<DbDataRecord>给匿名类型.虽然我们的这个集合不能进一步过滤,但是包含复杂类型的集合是可以被进一点过滤的.

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

Entity Framework 6 Recipes 2nd Edition(11-6)译 -> 从一个”模型定义”函数里返回一个复杂类型的更多相关文章

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

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

  2. Entity Framework 6 Recipes 2nd Edition(11-2)译 -> 为一个”模型定义”函数返回一个计算列

    11-3. 为一个”模型定义”函数返回一个计算列 问题 想从”模型定义”函数里返回一个计算列 解决方案 假设我们有一个员工(Employee)实体,属性有: FirstName, LastName,和 ...

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

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

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

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

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

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

  6. Entity Framework 6 Recipes 2nd Edition(9-7)译->在WCF服务中序列化代理

    9-7. 在WCF服务中序列化代理 问题 从一个查询里返回一个动态代理对象,想要把它序列为一个POCO(Plain-Old CLR Objects)对象. 实现基于POCO实体对象, 在运行时,EF会 ...

  7. Entity Framework 6 Recipes 2nd Edition(12-8)译 -> 重新获取一个属性的原始值

    12-8. 重新获取一个属性的原始值 问题 在实体保存到数据库之前,你想重新获取属性的原始值 解决方案 假设你有一个模型 (见 Figure 12-11) 表示一个员工( Employee),包含工资 ...

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

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

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

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

随机推荐

  1. 细说前端自动化打包工具--webpack

    背景 记得2004年的时候,互联网开发就是做网页,那时也没有前端和后端的区分,有时一个网站就是一些纯静态的html,通过链接组织在一起.用过Dreamweaver的都知道,做网页就像用word编辑文档 ...

  2. iOS---iOS10适配iOS当前所有系统的远程推送

    一.iOS推送通知简介 众所周知苹果的推送通知从iOS3开始出现, 每一年都会更新一些新的用法. 譬如iOS7出现的Silent remote notifications(远程静默推送), iOS8出 ...

  3. HTML 事件(三) 事件流与事件委托

    本篇主要介绍HTML DOM中的事件流和事件委托. 其他事件文章 1. HTML 事件(一) 事件的介绍 2. HTML 事件(二) 事件的注册与注销 3. HTML 事件(三) 事件流与事件委托 4 ...

  4. 学习ASP.NET Core, 怎能不了解请求处理管道[4]: 应用的入口——Startup

    一个ASP.NET Core应用被启动之后就具有了针对请求的处理能力,而这个能力是由管道赋予的,所以应用的启动同时意味着管道的成功构建.由于管道是由注册的服务器和若干中间件构成的,所以应用启动过程中一 ...

  5. angular2系列教程(九)Jsonp、URLSearchParams、中断选择数据流

    大家好,今天我们要讲的是http模块的第二部分,主要学习ng2中Jsonp.URLSearchParams.observable中断选择数据流的用法. 例子

  6. 操作系统篇-分段机制与GDT|LDT

    || 版权声明:本文为博主原创文章,未经博主允许不得转载. 一.前言     在<操作系统篇-浅谈实模式与保护模式>中提到了两种模式,我们说在操作系统中,其实大部分时间是待在保护模式中的. ...

  7. 如何利用pt-online-schema-change进行MySQL表的主键变更

    业务运行一段时间,发现原来的主键设置并不合理,这个时候,想变更主键.这种需求在实际生产中还是蛮多的. 下面,看看pt-online-schema-change解决这类问题的处理方式. 首先,创建一张测 ...

  8. PHP与JAVA构造函数的区别

    早期的PHP是没有面向对象功能的,但是随着PHP发展,从PHP4开始,也加入了面向对象.PHP的面向对象语法是从JAVA演化而来,很多地方类似,但是又发展出自己的特色.以构造函数来说,PHP4中与类同 ...

  9. TFS 测试用例步骤数据统计

    TFS系统集成了一套BI系统,基于SQL Server的Analysis Service进行实现的.通过这几年的深入使用,能够感触到这个数据数据仓库模型是多么的优秀,和微软官方提供的数据仓库示例Adv ...

  10. BPM配置故事之案例1-配置简单流程

    某天,Boss找到了信息部工程师小明. Boss:咱们新上了H3 BPM,你研究研究把现在的采购申请流程加上去吧,这是采购申请单. 小明:好嘞 采购申请单 小明回去后拿着表单想了想,开始着手配置. 他 ...