【转】Linq to EF 与Linq to Object 使用心得

大家都知道Linq既可以用来查询数据库对象(我这里指的是Entity FrameWork里的Model对象),也可以用来查询内存中的IEnumerable对象。

两者单独查询时都不会出现什么问题,不过混合在一起时(一般是用关键字来join连接),要注意的地方就多着了。

情形1:Linq to Object 连接(join) Linq to Entity

我们首先来看这段代码:(注意:Linq代码里是把内存中的数据代码,也就是Linq to object放在join前面,数据库的数据代码放在join后面)

[csharp] view plaincopy

  1. List<MyObject> objectList = new List<MyObject>();
  2. objectList.Add(new MyObject { Identity = 1, Name = "Jack", Age = 30 });
  3. objectList.Add(new MyObject { Identity = 2, Name = "Sam", Age = 28 });
  4. objectList.Add(new MyObject { Identity = 3, Name = "Lucy", Age = 23 });
  5. EntityRepository repository = new EntityRepository();
  6. DbSet<Entity> entitySet = repository.Context.Set<Entity>();
  7. var objectNames = (from ob in objectList
  8. join en in entitySet
  9. on ob.Identity equals en.SID
  10. select ob.Name).ToList();

Entity是数据库表,有一个bigint型,名为SID字段,其他俩个为Name和Notes。上面的代码中是把Linq to Object放在前面,Linq to Entity 放在join中,编译运行顺利。

实际的查询数据库的语句为:

[sql] view plaincopy

  1. SELECT
  2. [Extent1].[SID] AS [SID],
  3. [Extent1].[Name] AS [Name],
  4. [Extent1].[Notes] AS [Notes]
  5. FROM [dbo].[Entity] AS [Extent1]

这里是整表查询。(可以打开Sql Server 2008 R2-->Performance Tools-->SQL Server Profiler, 来跟踪此语句)。

如果把以上代码改成select en.Name,仍然是整表查询。

如果是设计两个数据库表的查询,且最后是select 表中的某一列,则不可能是整表查询,而是单独查询了某一列;而这里把entity和object放在一起,就会涉及整表查询。此乃两者混合使用的第一弊端。

情形2: Linq to Entity 连接(join) Linq to Object

数据库的数据代码Linq to Entity在join前,内存中数据代码Linq to Object在join后。代码如下:

[csharp] view plaincopy

  1. var entityNames = (from en in entitySet
  2. join ob in objectList
  3. on en.SID equals ob.Identity
  4. select en.Name).ToList();

好了,编译通过,运行时抛异常了。

Only Primitive types (‘Such as Int32, string, and Guid‘) are supported in this context

中文意思是“无法创建类型为“项目名.MyObject”的常量值。此上下文仅支持基元类型(“例如 Int32、String 和 Guid”)"

看来在涉及这种操作时,我们内存中的数据还不能是非基元类型。List<MyObject> objectList = new List<MyObject>();

MyObject要为int32, string或者Guid,才能运行通过,并且不是整表查询,而是针对name列的单独查询。大家可以一试。

所以在这里给出大家一点建议:

在涉及到内存中的对象与EF里的对象混合查询时,如果内存中的对象不为基元类型,则先把其中的某个要参加匹配的变量查询出来,再拿着它与EF对象混合查询。

这样不仅不会出错,效率也高,且代码会两段写,也容易看清楚意思。

以上的代码这样,才是最佳的

[csharp] view plaincopy

  1. IEnumerable<long> idList = objectList.Select(o => o.Identity);
  2. var entityNames = (from en in entitySet
  3. join id in idList
  4. on en.SID equals id
  5. select en.Name).ToList();

数据库查询语句如下:

[sql] view plaincopy

  1. SELECT
  2. [Extent1].[Name] AS [Name]
  3. FROM  [dbo].[Entity] AS [Extent1]
  4. INNER JOIN  (SELECT
  5. [UnionAll1].[C1] AS [C1]
  6. FROM  (SELECT
  7. cast(1 as bigint) AS [C1]
  8. FROM  ( SELECT 1 AS X ) AS [SingleRowTable1]
  9. UNION ALL
  10. SELECT
  11. cast(2 as bigint) AS [C1]
  12. FROM  ( SELECT 1 AS X ) AS [SingleRowTable2]) AS [UnionAll1]
  13. UNION ALL
  14. SELECT
  15. cast(3 as bigint) AS [C1]
  16. FROM  ( SELECT 1 AS X ) AS [SingleRowTable3]) AS [UnionAll2] ON  CAST( [Extent1].[SID] AS bigint) = [UnionAll2].[C1]

虽然里面的操作是麻烦了点,但是最后查询出来的东西却只有Name一个。相对而言,还是比较好的。

情形3:Linq to entities does not recognize the method ‘Int32 get_item(int32)‘

Linq to entities does not recognize the method ‘Int32 get_item(int32)‘ method, and this method cannot be translated into a store expression

Linq to Entities 不识别方法“Int64 get_Item(Int32)”,因此该方法无法转换为存储表达式。

且看代码

[csharp] view plaincopy

  1. List<long> list = new List<long> { 1, 2, 3 };
  2. var entity = entitySet.Where(en => en.SID == list[1]).ToList();

在这里数组的下标运算放在了linq表达式中,这种情况也是不被允许的。但是用contains,形如list.Contains(en.SID)

或者把下标操作放在linq语句都是可以的。

关系数据库不能识别 含有下标运算的表达式树翻译成的sql查询,为什么会这样呢? 也许要去问微软的工程师吧!

情形4:Only parameterless constructors and initializers are supported in LINQ to Entities

请看如下代码:

[csharp] view plaincopy

  1. EntityRepository repository = new EntityRepository();
  2. DbSet<Entity> entitySet = repository.Context.Set<Entity>();
  3. List<Tuple<long, string>> tuple = entitySet.Select(en => new Tuple<long, string>(en.SID, en.Name)).ToList();

仍然是针对EF的查询。这里的Tuple是元组类,可用多个参数构造成一个元组类(或叫匿名类)。运行时报异常,中文意思为:

LINQ to Entities 仅支持无参数构造函数和初始值设定项。 看来是在linq中使用了带参数的构造函数,linq是不支持这一点的。我们使用匿名类来试试看。

[csharp] view plaincopy

  1. var tuple = entitySet.Select(en => new { en.SID, en.Name }).ToList();

运行正常。但是有些情况下,我们非要使用Tuple来接受这两个参数以便于函数之间的传值,那该怎么办呢? 其实搞清楚真正的原因,自然就有办法了。因为我们在select后

.ToList()时要访问数据库,在这种情况下,带构造函数的Tuple自然是不能苟活的。所以只有在断开数据库后进行此操作才会万无一失。

[csharp] view plaincopy

  1. List<Tuple<long, string>> tuple = entitySet.Select(en => new { en.SID, en.Name })
  2. .AsEnumerable()
  3. .Select(en => new Tuple<long, string>(en.SID, en.Name)).ToList();

entitySet.Select(en => new { en.SID, en.Name })返回的是IQueryable类型,加了个AsEnumerable()之后便是IEnumerable类型,以此来断开数据库连接。

当然,用ToList()或ToArray()亦可!

时间: 2024-10-10 08:12:08

【转】Linq to EF 与Linq to Object 使用心得的相关文章

Linq to EF 和 Linq to SQL 中间Linq 产生的SQL语句

一,Linq to EF using System; using System.Runtime.CompilerServices; using System.Linq; using System.Data; namespace EntityframeworkSQL { class Program { static void Main(string[] args) { using (CustomerEntities customerEntities = new CustomerEntities()

EF与LINQ

1. EF跟LINQ不是一码事儿. 2. LINQ to EF是LINQ的一个provider,LINQ to SQL也是LINQ的一个provider.LINQ to EF是LINQ to SQL的替代产品,目前微软已经停止开发LINQ to SQL了,但是LINQ to SQL仍然有其用武之地. 3. LINQ是C#最优雅的特性,微软不可能放弃LINQ,除非开发出更优雅的可以替代LINQ的特性,否则放弃LINQ就等于放弃.NET.

应用开发之Linq和EF

上一章笔者对于WinForm开发过程用到的几个知识点做了讲解.笔者们可以以此为开端进行学习.而本章我们来讲一个跟ORM思想有关的知识点.在讲之前让我们想一下关于JAVA的hibernate知识点.hibernate也是ORM框架.记得hibernate里面有一个叫HQL.先不管HQL的好与坏.主要是明白HQL的目地是什么.ORM的思想就是为了让用户在操作数据的时候用上面向对象的思想来看,而不是二维数据了.所以HQL笔者认为就是一个面向对象思想的SQL语句.那么为什么笔者要讲到HQL呢?事实上笔者

Java进击C#——应用开发之Linq和EF

本章简言 上一章笔者对于WinForm开发过程用到的几个知识点做了讲解.笔者们可以以此为开端进行学习.而本章我们来讲一个跟ORM思想有关的知识点.在讲之前让我们想一下关于JAVA的hibernate知识点.hibernate也是ORM框架.记得hibernate里面有一个叫HQL.先不管HQL的好与坏.主要是明白HQL的目地是什么.ORM的思想就是为了让用户在操作数据的时候用上面向对象的思想来看,而不是二维数据了.所以HQL笔者认为就是一个面向对象思想的SQL语句.那么为什么笔者要讲到HQL呢?

Linq to EF 中Contains的演变

在早期Linq to EF中没有提供对Contains方法的支持, 那时候只能将所有数据获取到内存中,然后通过Linq to Object的Contains方法来达到相同的效果(如果多表筛选也可以使用Any方法实现,也可采用自定义linq的方式实现,但这里我们主要讨论使用数组筛选表的情况). 从Linq to EF 4.0 开始加入了对Contains的支持,使用方式如下: 以上linq最终会被翻译为如下形式的sql语句: SELECT [Extent1].[Name] AS [Name] FR

Linq技术三:Linq to Object 和生成数据表的扩展方法

这篇来谈论一下Linq第三个方面的应用:Linq to Object,只要是继承了IEnumerable或IQueryable接口的Object都能使用Linq特性进行操作.在操作过程当中可能很多人都觉得不好调试不能实时地观察结果数据集,想把IQuery的Linq查询语句转换成数据表DataTable,要怎么实现转换呢?来看一下. 先来说一场景解释一下为什么需要用Linq来解决一些问题,能解决一些什么样的问题,相对于SQL,DataTable等一些传统操作方式有哪些优势? 场景:目前主要数据源有

linq和EF查询的用法和区分

我们做项目时,难免会遇到用的不知道是啥,及把linq和EF搞混了 今天我带领大家梳理下思路: 首先说linq查询,然后介绍EF查询 1.linq查询 当我们使用linq查询时,转到定义会调到Queryable 类,  那么也就是说,这个类封装了linq所有查询的方法,那么我们来研究研究这个类 MSDN上是这样解释的:提供一组用于查询实现 IQueryable<T> 的数据结构的 static(在 Visual Basic 中为 Shared)方法. 命名空间:   System.Linq程序集

linq to ef(相当于sql中in的用法)查询语句

select * from DoctorInfo doctor where doctor.HosDepartId in (select Id from HospitalDepartment hd where hd.DepartmentId=5) var a=from d in _entity.HospitalDepartment where d.DepartmentId==5 select d; List<int> lst=new List<int>();foreach(var b

Linq To SQL和Linq To Object的批量操作InsertAllOnSubmit介绍

无论是Linq To SQL还是Linq To Object(Entity frameworks)它们都为开发人员提供了Insert操作,及Insert集合操作,即InsertOnSubmit和InsertAllOnSubmit,前者是将一个实体标记为一个插入状态,而后都是将一个集合标记为插入状态,... 无论是Linq To SQL还是Linq To Object(Entity frameworks)它们都为开发人员提供了Insert操作,及Insert集合操作,即InsertOnSubmit