最近工作中遇到数据库组合查询带来的一些问题,因此有必要调研一下Linq to Objects Join Linq to Entity。参考一些网友的代码案例,深入实践了一下使用EntityFramework Code First 下的组合查询。

准备环节:

  (一) 在VS下创建一个控制台应用程序(LinqToObjectJoinEntity),定义一个MyObject类,如下:

    public class MyObject
    {  
      public int Identity { get; set; }
      public string Name { get; set; }
      public int Age { get; set; }
    }

  (二)然后再定义一个Entity类及EntityContext类。 

    public class Entity
    {
      public int EntityId { get; set; }
      public string Name { get; set; }
      public string Notes { get; set; }
    }

    public class EntityContext : DbContext
    {
      public IDbSet<Entity> Entitys { get; set; }

      protected override void OnModelCreating(DbModelBuilder modelBuilder)
      {
        modelBuilder.Conventions.Remove<System.Data.Entity.ModelConfiguration.Conventions.PluralizingTableNameConvention>();      // 此处是避免使用EF创建出来的Entity表为复数形式,顺便吐槽下微软默认给复数的设计
        modelBuilder.Conventions.Remove<System.Data.Entity.ModelConfiguration.Conventions.OneToManyCascadeDeleteConvention>();   //  此处是避免使用EF创建出的数据库包含数据迁移表

        base.OnModelCreating(modelBuilder);
      }
    }

  (三)使用EF创建准备数据(在Main()函数中实现):

    using (var db = new EntityContext())
    {
      db.Entitys.Add(new Entity { EntityId = 1, Name = "Entity", Notes = "Notes" });
      db.Entitys.Add(new Entity { EntityId = 2, Name = "Frame", Notes = "Mates" });
      db.Entitys.Add(new Entity { EntityId = 3, Name = "Work", Notes = "Honor" });

      db.SaveChanges();
    }

Coding实践:

  (1)重现Linq to Object Join Linq to Entity

    #region Reproduce Linq to Object Join Linq to Entity

    var objectNames = (from myObject in myObjects
             join entity in db.Entitys
             on myObject.Identity equals entity.EntityId
             select myObject.Name).ToList();

    #endregion

    使用Sql Profiler观察到的查询语句如下:

    SELECT
    [Extent1].[EntityId] AS [EntityId],
    [Extent1].[Name] AS [Name],
    [Extent1].[Notes] AS [Notes]
    FROM [dbo].[Entity] AS [Extent1]

    属于全表查询,此乃Linq to objects Join Linq to Entity一大弊害。

  (2)重现 Linq to Entity Join Linq to Object

    #region Reproduce Linq to Entity Join Linq to Object

    var entityName = (from entity in db.Entitys
    join myObject in myObjects
    on entity.EntityId equals myObject.Identity
    select entity.Name).ToList();

    #endregion

    这个地方运行时会抛异常:

      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列的单独查询。

  (3)改进 Linq to Entity Join Linq to Object

    #region Linq to Entity Join Linq to Object(Resolve)

    var identities = myObjects.Select(o => o.Identity);

    var entitytNames = (from entity in db.Entitys
             join identity in identities
             on entity.EntityId equals identity
             select entity.Name).ToList();

    #endregion

    这里Sql Profile监测到的查询语句为:

    SELECT
    [Extent1].[Name] AS [Name]
    FROM [dbo].[Entity] AS [Extent1]
    INNER JOIN (SELECT
    1 AS [C1]
    FROM ( SELECT 1 AS X ) AS [SingleRowTable1]
    UNION ALL
      SELECT
      2 AS [C1]
      FROM ( SELECT 1 AS X ) AS [SingleRowTable2]
    UNION ALL
      SELECT
      3 AS [C1]
      FROM ( SELECT 1 AS X ) AS [SingleRowTable3]) AS [UnionAll2] ON [Extent1].[EntityId] = [UnionAll2].[C1]

    虽是麻烦了些,查出来的东西只有一个。

此文只是针对性的简述下Linq to Object Join Linq to Entity的场景,在日常工作中可能不止于此。

代码见:https://github.com/Jashinck/LinqToObjectJoinEntity

深入调研Linq to Objects Join Linq to Entity的更多相关文章

  1. Linq之旅:Linq入门详解(Linq to Objects)

    示例代码下载:Linq之旅:Linq入门详解(Linq to Objects) 本博文详细介绍 .NET 3.5 中引入的重要功能:Language Integrated Query(LINQ,语言集 ...

  2. LINQ之LINQ to Objects(上)

    LINQ概述 LINQ,语言集成查询(Language Integrated Query),它允许使用C#或VB代码以查询数据库相同的方式来操作不同的数据源. 1.LINQ体系结构 从上图可以看出,L ...

  3. 从LINQ开始之LINQ to Objects(上)

    LINQ概述 LINQ,语言集成查询(Language Integrated Query),它允许使用C#或VB代码以查询数据库相同的方式来操作不同的数据源. LINQ体系结构 从上图可以看出,LIN ...

  4. Linq之旅:Linq入门详解(Linq to Objects)【转】

    http://www.cnblogs.com/heyuquan/p/Linq-to-Objects.html Linq之旅:Linq入门详解(Linq to Objects) 示例代码下载:Linq之 ...

  5. Linq之旅:Linq入门详解(Linq to Objects)(转)

    http://www.cnblogs.com/heyuquan/p/Linq-to-Objects.html 示例代码下载:Linq之旅:Linq入门详解(Linq to Objects) 本博文详细 ...

  6. Linq之Linq to Objects

    目录 写在前面 系列文章 linq to objects 总结 写在前面 上篇文章介绍了linq的延迟加载特性的相关内容,从这篇文章开始将陆续介绍linq to Objects,linq to xml ...

  7. LINQ to Objects系列(1)相关技术准备

    LINQ to Objects是LINQ的一部分,是查询对象集合的一种语法.首先看一下LINQ的体系结构,这样对LINQ有一个大致的了解.如图. 第一篇文章主要是回顾一下学习LINQ to Objec ...

  8. 编写高质量代码改善C#程序的157个建议[IEnumerable<T>和IQueryable<T>、LINQ避免迭代、LINQ替代迭代]

    前言 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议29.区别LINQ查询中的IEnumerable<T ...

  9. Linq学习(一)-初涉Linq

    一.何谓LINQ LINQ:Language Integrated Query语言集成查询,其本质是对ADO.NET结果集通过反射连同泛型特性转换成对象集,实现OR模型的转换 二.优点与缺点 优点:封 ...

随机推荐

  1. 从头认识js-js中的继承

    要彻底弄明白js中的继承,我们首先要弄清楚js中的一个很重要的概念那就是原型链. 1.什么是原型链? 我们知道每个构造函数都有一个原型对象,原型对象包含一个指向构造函数的指针,而实例都包含一个指向原型 ...

  2. AWS EC2+Docker+JMeter构建分布式负载测试基础架构

    目录 概述及范围 前提条件 Part 1: Local setup-本地配置 Part 2: Cloud端基础架构--Infrastructure 总结: 原文链接 @ 概述及范围 本文介绍有关如何使 ...

  3. Vue+Axios+iview+vue-router实战管理系统项目

    项目实现登陆验证,iviewui组件使用,vue-router路由拦截,在删除登陆存储的token后,直接路由进登陆页面,实战数据请求,本地data.json,笔者也是初出茅庐,在项目运行后,会一直c ...

  4. git版本回退问题记录

    因为之前有个前端改了文件目录进行合并时候丢失掉些许代码,然后我在以前分支进行了代码层级的整理,项目如果想要启动还需还原回以前的版本,我进行了三次文件夹层级提交,所以我需要进行三次的版本回退. git命 ...

  5. 基于springcloud搭建项目-Feign篇(四)

    上一篇已经写过ribbon客户端负载均衡的用法了,这篇主要是介绍feign的用法,首先我们必须了解feign是什么?能干嘛?怎么用? 这里简单介绍一下,然后进行代码测试 1.概述 Feign是一个声明 ...

  6. IRM3800 红外遥控器解码 linux驱动

    这一次还是接在 Cemera 上.用 中断引脚 EINT20 也就是 GPG12. 之前焊的 51 板子上有一个红外接收器. 请注意了,是 标准的 NEC 码规范:首次发送的是9ms的高电平脉冲,其后 ...

  7. JAVA生成EXCEL模板

    JAVA生成excel模板,支持1.必填字段前加 红色 * 2.定义可选值下拉列表 valList3.定义名称并通过名称设置可选值 refName(名称在sheet2,sheet2自动隐藏)4.支持设 ...

  8. CSS--transform相关属性实现2d到3d的具体变化

    先放上一段我写的相关代码(可能有一定冗杂代码,请见谅) <!DOCTYPE html> <html lang="en"> <head>      ...

  9. 04-influxdb基本操作

    influxdb基本操作 1. 数据库基本操作 # 创建数据库 > create database db01; # 查看数据库 > show databases; name: databa ...

  10. python sqlite3操作类扩展,包含数据库分页

     一.原因 最近在使用python3和sqlite3编辑一些小程序,由于要使用数据库,就离不开增.删.改.查,sqlite3的操作同java里的jdbc很像,于是就想找现成的操作类,找来找去,发现一个 ...