Entity Framework 6 Recipes 2nd Edition(11-5)译 -> 从”模型定义”函数返回一个匿名类型
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)译 -> 从”模型定义”函数返回一个匿名类型的更多相关文章
- Entity Framework 6 Recipes 2nd Edition(11-1)译 -> 从“模型定义”函数返回一个标量值
第11章函数 函数提供了一个有力代码复用机制, 并且让你的代码保持简洁和易懂. 它们同样也是EF运行时能利用的数据库层代码.函数有几类: Rowset Functions, 聚合函数, Ranking ...
- Entity Framework 6 Recipes 2nd Edition 译 -> 目录 -持续更新
因为看了<Entity Framework 6 Recipes 2nd Edition>这本书前面8章的翻译,感谢china_fucan. 从第九章开始,我是边看边译的,没有通读,加之英语 ...
- Entity Framework 6 Recipes 2nd Edition(11-2)译 -> 为一个”模型定义”函数返回一个计算列
11-3. 为一个”模型定义”函数返回一个计算列 问题 想从”模型定义”函数里返回一个计算列 解决方案 假设我们有一个员工(Employee)实体,属性有: FirstName, LastName,和 ...
- Entity Framework 6 Recipes 2nd Edition(10-4)译 -> 从存储过程返回一个复杂类型
10-4. 从存储过程返回一个复杂类型 问题 想在方法中使用一个返回复杂类型的存储过程 解决方案 假设我们已经有如Figure 10-3.所示的模型,该Employee (雇员)模型包含Employe ...
- Entity Framework 6 Recipes 2nd Edition(9-3)译->找出Web API中发生了什么变化
9-3. 找出Web API中发生了什么变化 问题 想通过基于REST的Web API服务对数据库进行插入,删除和修改对象图,而不必为每个实体类编写单独的更新方法. 此外, 用EF6的Code Fri ...
- Entity Framework 6 Recipes 2nd Edition(9-4)译->Web API 的客户端实现修改跟踪
9-4. Web API 的客户端实现修改跟踪 问题 我们想通过客户端更新实体类,调用基于REST的Web API 服务实现把一个对象图的插入.删除和修改等数据库操作.此外, 我们想通过EF6的Cod ...
- Entity Framework 6 Recipes 2nd Edition(目录索引)
Chapter01. Getting Started with Entity Framework / 实体框架入门 1-1. A Brief Tour of the Entity Framework ...
- Entity Framework 6 Recipes 2nd Edition(9-1)译->用Web Api更新单独分离的实体
第九章 在N层结构的应用程序中使用EF 不是所有的应用都能完全地写入到一个单个的过程中(就是驻留在一个单一的物理层中),实际上,在当今不断发展的网络世界,大量的应用程序的结构包含经典的表现层,应用程, ...
- Entity Framework 6 Recipes 2nd Edition(13-1)译 -> 优化TPT继承模型的查询
问题 你想提高在一个TPT继承模型里的查询 解决方案 让我们假设有一个简单的TPT继承模型,如图Figure 13-1 Figure 13-1. A simple Table per Type inh ...
随机推荐
- node-webkit 环境搭建与基础demo
首先去github上面下载(地址),具体更具自己的系统,我的是windows,这里只给出windows的做法 下载windows x64版本 下载之后解压,得到以下东西 为了方便,我们直接在这个目录中 ...
- 动画requestAnimationFrame
前言 在研究canvas的2D pixi.js库的时候,其动画的刷新都用requestAnimationFrame替代了setTimeout 或 setInterval 但是jQuery中还是采用了s ...
- 使用 Roslyn 编译器服务
.NET Core和 .NET 4.6中 的C# 6/7 中的编译器Roslyn 一个重要的特性就是"Compiler as a Service",简单的讲,就是就是将编译器开放为 ...
- input[tyle="file"]样式修改及上传文件名显示
默认的上传样式我们总觉得不太好看,根据需求总想改成和上下结构统一的风格…… 实现方法和思路: 1.在input元素外加a超链接标签 2.给a标签设置按钮样式 3.设置input[type='file' ...
- UWP开发之Template10实践:本地文件与照相机文件操作的MVVM实例(图文付原代码)
前面[UWP开发之Mvvmlight实践五:SuspensionManager中断挂起以及复原处理]章节已经提到过Template10,为了认识MvvmLight的区别特做了此实例. 原代码地址:ht ...
- H5坦克大战之【画出坦克】
今天是个特殊的日子,圣诞节,也是周末,在这里先祝大家圣诞快乐!喜庆的日子,我们可以稍微放松一下,扯一扯昨天雷霆对战凯尔特人的比赛,这场比赛大威少又双叒叕拿下三双,而且是一个45+11+11的超级三双, ...
- 如何避免git每次提交都输入密码
在ubuntu系统中,如何避免git每次提交都输入用户名和密码?操作步聚如下:1: cd 回车: 进入当前用户目录下:2: vim .git-credentials (如果没有安装vim 用其它编辑器 ...
- PHP设计模式(六)原型模式(Prototype For PHP)
原型设计模式: 用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 原型设计模式简单的来说,顾名思义, 不去创建新的对象进而保留原型的一种设计模式. 缺点:原型设计模式是的最主要的缺点就 ...
- Salesforce开发者学习笔记之一:基本知识
本文介绍了Salesforce开发平台的基本知识, 包括如下内容: Salesforce平台介绍 Salesforce基本术语 定制和扩展Salesforce平台 创建一个简单的应用程序 Salesf ...
- 札记:Java异常处理
异常概述 程序在运行中总会面临一些"意外"情况,良好的代码需要对它们进行预防和处理.大致来说,这些意外情况分三类: 交互输入 用户以非预期的方式使用程序,比如非法输入,不正当的操作 ...