11-5. 从”模型定义”函数返回一个匿名类型

问题

想创建一个返回一个匿名类型的”模型定义”函数

解决方案

假设已有游客(Visitor) 预订(reservation)房间(hotel ) 的模型,如Figure 11-5所示.

Figure 11-5. A model for hotel reservations

想要返回每位游客房间预订条数和带来的总收入.因为很多地方需要这些信息,所以想要创建一个”模型定义”函数,接受一个查询参数,返回一个包含游客合计信息的匿名类型的集合:

2. 把Listing 11-9 中的代码,插入到.edmx文件的概念模型里<Schema> 标签下,这样我们就定义好了函数.

Listing 11-9. The VisitorSummary() Model-Defined Function

<Function Name="VisitorSummary">

<Parameter Name="StartDate" Type="Edm.DateTime" />

<Parameter Name="Days" Type="Edm.Int32" />

<ReturnType>

<CollectionType>

<RowType>

<Property Name="Name" Type="Edm.String" />

<Property Name="TotalReservations" Type="Edm.Int32" />

<Property Name="BusinessEarned" Type="Edm.Decimal" />

</RowType>

</CollectionType>

</ReturnType>

<DefiningExpression>

Select

r.Visitor.Name,

COUNT(r.ReservationId) as TotalReservations,

SUM(r.Cost) as BusinessEarned

from EFRecipesEntities.Reservations as r

where r.ReservationDate between StartDate and

AddDays(StartDate,Days)

group by r.Visitor.Name

</DefiningExpression>

</Function>

3. 接下来插入和查询模型,代码如 Listing 11-10所示:.

Listing 11-10. Querying the Model Using the VistorySummary() Model-Defined Function

class Program

{

static void Main(string[] args)

{

RunExample();

}

static void RunExample()

{

using (var context = new EFRecipesEntities())

{

var hotel = new Hotel { Name = "Five Seasons Resort" };

var v1 = new Visitor { Name = "Alex Stevens" };

var v2 = new Visitor { Name = "Joan Hills" };

var r1 = new Reservation

{

Cost = 79.99M,

Hotel = hotel,

ReservationDate = DateTime.Parse("2/19/2010"),

Visitor = v1

};

var r2 = new Reservation

{

Cost = 99.99M,

Hotel = hotel,

ReservationDate = DateTime.Parse("2/17/2010"),

Visitor = v2

};

var r3 = new Reservation

{

Cost = 109.99M,

Hotel = hotel,

ReservationDate = DateTime.Parse("2/18/2010"),

Visitor = v1

};

var r4 = new Reservation

{

Cost = 89.99M,

Hotel = hotel,

ReservationDate = DateTime.Parse("2/17/2010"),

Visitor = v2

};

context.Hotels.Add(hotel);

context.SaveChanges();

}

using (var context = new EFRecipesEntities())

{

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

var esql = @"Select value v from

EFRecipesModel.VisitorSummary(DATETIME'2010-02-16 00:00', 7) as v";

var objectContext = (context as IObjectContextAdapter).ObjectContext;

var visitors = objectContext.CreateQuery<DbDataRecord>(esql);

foreach (var visitor in visitors)

{

Console.WriteLine("{0}, Total Reservations: {1}, Revenue: {2:C}",

visitor["Name"], visitor["TotalReservations"],

visitor["BusinessEarned"]);

}

}

using (var context = new EFRecipesEntities())

{

Console.WriteLine();

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

//译注:在我的EF6.1.1.3下,这句查询会出现异常

var visitors = from v in

context.VisitorSummary(DateTime.Parse("2/16/2010"), 7)

select v;

foreach (var visitor in visitors)

{

Console.WriteLine("{0}, Total Reservations: {1}, Revenue: {2:C}",

visitor["Name"], visitor["TotalReservations"],

visitor["BusinessEarned"]);

}

}

}

}

partial class EFRecipesEntities

{

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

public IQueryable<DbDataRecord> VisitorSummary(DateTime StartDate, int Days)

{

var objectContext = (this as IObjectContextAdapter).ObjectContext;

return objectContext.CreateQuery<DbDataRecord>(

Expression.Call(Expression.Constant(this),

(MethodInfo)MethodInfo.GetCurrentMethod(),

new Expression[] { Expression.Constant(StartDate),

Expression.Constant(Days) }

).ToString());

}

}

上面Listing 11-10 的代码输出如下:

Using eSQL...

Alex Stevens, Total Reservations: 2, Revenue: $189.98

Joan Hills, Total Reservations: 2, Revenue: $189.98

Using LINQ...

Alex Stevens, Total Reservations: 2, Revenue: $189.98

Joan Hills, Total Reservations: 2, Revenue: $189.98

它是如何工作的?

在Listing 11-9里, 在VisitorSummary() 函数定义里,我们按实体的导航属性visitor进行分组,我们用eSQL Count() 的 函数计算每位游客的预订房间数量,用Sum() 把每位游客付的租金合计.

在这个函数里,我们把结果组织成:: Name, TotalReservations, 和BusinessEarned. 此处我们使用了<CollectionType> 和<RowType> 标签来指明返回类型. 在运行时里,用包含DbDataRecords的集合

为了使该函数能在LINQ查询中使用,我们创建了运行时方法(返回IQueryable<DbDataRecord>.) .和前面的小节一样,我们用EdmFunction()特性修饰该方法.. 因为需要返回一个IQueryable<T>, 所以我们需要在这个方法体中包含函数调用,以便它能用在查询表达式树中.

此外,我们需要在ObjectContext中访问QueryProvider来返回一个IQueryable<T>,所以我们需要在EFRecipesEntities类中实现这个方法.

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

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

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

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

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

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

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

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

  4. Entity Framework 6 Recipes 2nd Edition(10-4)译 -> 从存储过程返回一个复杂类型

    10-4. 从存储过程返回一个复杂类型 问题 想在方法中使用一个返回复杂类型的存储过程 解决方案 假设我们已经有如Figure 10-3.所示的模型,该Employee (雇员)模型包含Employe ...

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

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

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

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

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

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

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

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

  9. Entity Framework 6 Recipes 2nd Edition(13-1)译 -> 优化TPT继承模型的查询

    问题 你想提高在一个TPT继承模型里的查询 解决方案 让我们假设有一个简单的TPT继承模型,如图Figure 13-1 Figure 13-1. A simple Table per Type inh ...

随机推荐

  1. Python高手之路【二】python基本数据类型

    一:数字 int int(整型): 在32位机器上,整数的位数为32位,取值范围为-2**31-2**31-1,即-2147483648-2147483647 在64位系统上,整数的位数为64位,取值 ...

  2. ASP.NET MVC5+EF6+EasyUI 后台管理系统(74)-微信公众平台开发-自定义菜单

    系列目录 引言 1.如果不借用Senparc.Weixin SDK自定义菜单,编码起来,工作量是非常之大 2.但是借助SDK似乎一切都是简单得不要不要的 3.自定义菜单无需要建立数据库表 4.自定义菜 ...

  3. jQuery学习之路(7)- 用原生JavaScript实现jQuery的某些简单功能

    ▓▓▓▓▓▓ 大致介绍 学习了妙味,用原生的JavaScript实现jQuery中的某些部分功能 定义自己的函数库lQuery ▓▓▓▓▓▓ $()选择器的实现 jQuery是面向对象的,所以自己编写 ...

  4. [C#] 走进异步编程的世界 - 开始接触 async/await

    走进异步编程的世界 - 开始接触 async/await 序 这是学习异步编程的入门篇. 涉及 C# 5.0 引入的 async/await,但在控制台输出示例时经常会采用 C# 6.0 的 $&qu ...

  5. CRL快速开发框架系列教程六(分布式缓存解决方案)

    本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...

  6. 玩转spring boot——结合JPA入门

    参考官方例子:https://spring.io/guides/gs/accessing-data-jpa/ 接着上篇内容 一.小试牛刀 创建maven项目后,修改pom.xml文件 <proj ...

  7. [systemtap手记]debian体系安装过程

    Debian体系,本人测试用机 Ubuntu 11.10 uname -r 查看原本的内核版本为 3.0.0-12-generic 第一步: 安装systemtap包 $ sudo apt-get i ...

  8. python 3.5 成功安装 scrapy 的步骤

    http://www.cnblogs.com/hhh5460/p/5814275.html

  9. 【夯实PHP基础】PHP常用类和函数总结

    本文地址 代码提纲: 1. 字符串处理类及函数 2. 数组处理类及函数 3 .web处理类及函数 将常用的PHP的类和函数总结到这里,主要是 自己用过的,比较有感觉. 1. [字符串处理] 1)[ut ...

  10. mysql删除重复记录语句的方法

    例如: id name value 1 a pp 2 a pp 3 b iii 4 b pp 5 b pp 6 c pp 7 c pp 8 c iii id是主键 要求得到这样的结果 id name ...