Entity Framework 4.0 recipes 读书笔记2 ExecuteStoreQuery()

写在之前:我想通过refector 反编译一下system.data.entity.dll(4.0 version),发现反编译出来的只有属性申明和方法声明,里面一句代码都没有,真是火大啊,试了几个.net4.0的其它dll同样也是如此,并且我的机器上的refector是最新版的,后来我觉得可能是dll的路径不对,我引用的是C:\Program Files\Reference Assemblies\Microsoft\Framework\.NETFramework\v4.0\Profile\Client\System.Data.Entity.dll,后来查找才知道,正确的路径是C:\Windows\Microsoft.NET\Framework\v4.0.30319\System.Data.Entity.dll。不过很奇怪的是这个路径下C:\Program Files\Reference Assemblies\Microsoft\Framework\.NETFramework\v3.5\Profile\Client\System.Data.Entity.dll反编译居然不正常。

对一个entity data model进行查询的方式除了用linq to entity,linq to object,esq,还可以用sql ,store procedure(我有做写过blog,借助EFExtension和EntityClient来实现).不过在EF1中使用sql,procedure返回entity 是比较麻烦的。不过在EF4中对于使用sql ,store procedure返回entity就方便多了,因为不用再去调用EntityClient了,因为objectcontext提供了ExecuteStoreQuery<TElement>(),所以还是说具体的情况:

1.Executing SQL Statement:如果需要直接去数据库表查询,可直接使用object context的ExecuteStoreCommand()方法。

假设数据库表payment (PaymentId,Amount,Vendor),对应entity data model有一个Payment对象,

下面的代码片段是通过object context insert payment,记得在EF1.0的是时候完成这个操作需要借助EntityClient.

图1

using (var context = new EFRecipesEntities())
           {
               string sql = @"insert into Payment(Amount, Vendor)
                              values (@Amount, @Vendor)";
               var args = new DbParameter[] {
                   new SqlParameter { ParameterName = "Amount", Value = 99.97M},
                   new SqlParameter { ParameterName = "Vendor", Value="Ace Plumbing"}
               };
               int rowCount = context.ExecuteStoreCommand(sql, args);

args = new DbParameter[] {
                   new SqlParameter { ParameterName = "Amount", Value = 43.83M},
                   new SqlParameter { ParameterName = "Vendor", Value = "Joe‘s Trash Service"}
               };
               rowCount += context.ExecuteStoreCommand(sql, args);
               Console.WriteLine("{0} rows inserted", rowCount.ToString());
           }

ExecuteStoreCommand()返回一个int值,影响的行数。

2.Returning Objects from a SQL Statement:通过sql查询返回object实体,这个在EF1.0我也做过练习,那时候是通过EFExtension来实现的:

下面的例子

using (var context = new EFRecipesEntities())
{
string sql = "select * from  Payment where Vendor= @vendor";
var args = new DbParameter[] {
new SqlParameter {ParameterName = "Vendor", Value = "ken"}};
var students = context.ExecuteStoreQuery<Payment >(sql, args);
}

这是一个非常简单的例子,对于ExecuteStoreQuery()有有许多需要注意:

1.sql = "select * from  Payment where Vendor= @vendor";之所以能写成select *是因为Payment对象的属性和表的字段命名完全一致,如果不一致的话,需要将表字段 取别名,别名需是对象映射的属性名称。

2.如果sql语句返回的列少于(具体化)实体的属性的个数,那么EF在具体化的时候将抛出一个异常如下图,因此将需要缺少的列补上一些没有意义的值,以保证在具体乎的时候不会报错:eg 如图1,如果sql=”select PaymentId ,Amount from Payment ” 这样使用context.ExecuteStoreQuery<Payment >(sql, args);那么会报异常,因此需要将Vendor 列补上 。

正确的sql=”select PaymentId ,Amount, null as Vendor from Payment”。

3.如果sql 返回的列 多余具体化的实体属性的个数,那么EF将会忽视多出的列。

如下的代码调试可以通过:多出的test列在具体化(materialization)的时候是会忽略掉的

string sql = "select PaymentId, Amount,Vendor,null as test from Chapter3.Payment “;

var students = context.ExecuteStoreQuery<Payment>(sql); ;
     Console.WriteLine("Payment...");

4.如果是你返回的表是映射到几个继承关系的实体类上,那么返回的行需要具体化到几个实体上,EF是无法根据识别列来将返回的行具体化到相应的继承类型上去,这是EF会抛出一个运行时的exception


图1


   图2

sing (var context = new EFRecipesEntities())
{
    string sql = "select * from Employee";
    var employee = context.ExecuteStoreQuery<Employee>(sql);                  
}

如上图模型和代码,图2是异常信息。

5.如果实体有complex Type属性,那么实体对象的实例是无法用ExecuteStoreQuery()来返回的,因为ExcuteStoreQuery()是无法返回一个complex Type的集合的.返回单个complex type是支持的,但是返回的实体对象里包含complex type就不支持。

6.可以返回实体对象属性的子集,就是说如果对于Payment表,我们查询返回PaymentId和Amount字段,然后我们定义一个subPayment 实体包含PaymentId和Amount属性,然后使用ExcuteStoreQuery<subPayment>()

3.Returning Objects from an Entity SQL Statement:通过esql返回实体对象. 这个和EF1.0没什么区别,可以去下载最新版的EF query sample

4.setting a default value in a query 在查询中设置默认值:

1.最简单的办法就是在实体的属性上设置默认值

2.就是使用匿名实体,就像这样 var employees = from e in context.Employees select new {Name = e.Name, YearsWorked = e.YearsWorked ?? 0};

3.通过sql语句,不过这需要借助DbDataRecord来实现,eg:string esql = @"select e.Name,case  when .YearsWorked is null then 0 else e.YearsWorked end as YearsWorked from Employees as e"; var employees = context.CreateQuery<DbDataRecord>(esql);

4.对于linq to entity没有相应的方法。

时间: 2024-11-05 19:32:42

Entity Framework 4.0 recipes 读书笔记2 ExecuteStoreQuery()的相关文章

Entity Framework 4 in Action读书笔记——第一章:数据访问重载:Entity Fram

小分享:我有几张阿里云优惠券,用券购买或者升级阿里云相应产品最多可以优惠五折!领券地址:https://promotion.aliyun.com/ntms/act/ambassador/sharetouser.html?userCode=ohmepe03 写在之前的话 在深入研究实体框架的细节之前,我们先讨论从传统的DataSet方法转换到基于对象的方法实现数据访问所带来的便利,以及这两种方法不同的工作方式是怎样导致采用像Entity Framework这样的O/RM工具. 使用Dataset和

初次开发 ASP.NET vNext 续篇:云优化的概念、Entity Framework 7.0、目前性能还不够好

继续上一篇<开发 ASP.NET vNext 初步总结(使用Visual Studio 2014 CTP1)>之后, 关于云优化和版本控制: 我本想做一下MAC和LINUX的self-host测试,但是官方说运行环境的MONO版本至少需要3.4.1,我去年买了个表,至本文发布为止,你让我下地狱去找3.4.1吗,硬着头皮用3.4.0搞了一晚上,MAC一直停留在 httpapi.dll出错,Ubuntu Server 12.0.4 是不认其中的几个DLL包,具体哪几个也忘了,过段时间有了稳定版本再

Entity Framework 5.0 Code First全面学习

目录(?)[+] 不贴图片了,太累. Code First 约定 借助 CodeFirst,可通过使用 C# 或Visual Basic .NET 类来描述模型.模型的基本形状可通过约定来检测.约定是规则集,用于在使用 Code First 时基于类定义自动配置概念模型.约定是在 System.Data.Entity.ModelConfiguration.Conventions 命名空间中定义的. 可通过使用数据注释或Fluent API 进一步配置模型.优先级是通过 Fluent API 进行

Entity Framework 6.0 对枚举的支持/实体添加后会有主键反回

实验 直接上代码,看结果 实体类 [Flags] public enum FlagsEnum { Day = 1, Night = 2 } public class EntityWithEnum { public int ID { get; set; } public FlagsEnum ValidTime { get; set; } } 数据库上下文 public partial class CodeFirstModel : DbContext { public CodeFirstModel(

Entity Framework 6.0 Tutorials(1):Introduction

以下系统文章为EF6.0知识的介绍,本章是第一篇 原文地址:http://www.entityframeworktutorial.net/entityframework6/introduction.aspx ------------------------------------------------------------------------------------------------------------- Entity Framework 6.0 Introduction: W

Entity Framework 5.0 Code First全面学习 (转)

原文地址:感谢原文作者 http://blog.csdn.net/gentle_wolf/article/details/14004345 不贴图片了,太累. Code First 约定 借助 CodeFirst,可通过使用 C# 或Visual Basic .NET 类来描述模型.模型的基本形状可通过约定来检测.约定是规则集,用于在使用 Code First 时基于类定义自动配置概念模型.约定是在 System.Data.Entity.ModelConfiguration.Convention

ORM Entities vs. Domain Entities under Entity Framework 6.0

I stumbled upon the following two articles First and Second in which the author states in summary that ORM Entities and Domain Entities shouldn't be mixed up. I faced exactly this problem at the moment as I code with EF 6.0 following the Code First a

Entity Framework 5.0系列之自动生成Code First代码

在前面的文章中我们提到Entity Framework的“Code First”模式也同样可以基于现有数据库进行开发.今天就让我们一起看一下使用Entity Framework Power Tools如何基于现有数据库生成数据类和数据库上下等. Entity Framework Power Tools 基于现有数据库生成POCO数据类和数据库上下文需要借助Visual Studio一个扩展插件-- Entity Framework Power Tools(一个Code First反向工程工具).

Entity Framework 5.0.0 Function Import 以及 Implicit REF CURSOR Binding

源代码 概要:1,明如何使用Entity Framework中的function import功能. 2,说明如何使用ODP.NET的隐式REF CURSOR绑定(implicit REF CURSOR binding). 环境以及工具: Windows 10 企业版 Microsoft Visual Studio Enterprise 2015 Oracle Database 11g Enterprise Edition Release 11.2.0.1.0 .NET Framework 4.