重学c#系列——linq(3) [二十九]
前言
继续介绍一些复杂的linq。
正文
groupjoin 这个函数:
有department
public class Deployment
{
public string Id { get; set; }
public Deployment(string id)
{
Id = id;
}
}
有Employee:
public class Employee
{
public string DepartmentId { get; set; }
public string Name { get; set; }
public Employee(string name, string deploymentId)
{
Name = name;
DepartmentId = deploymentId;
}
}
现在要实现Deployment和Employee,一对多的关系:
List<Deployment> a = new List<Deployment>()
{
new Deployment("1"),
new Deployment("2"),
};
List<Employee> e = new List<Employee>()
{
new Employee("张三","1"),
new Employee("李四","1"),
new Employee("王五","2"),
};
如果我们使用join的话,那么就是:
Deployment("1") Employee("张三","1")
Deployment("1") Employee("李四","1")
Deployment("2") Employee("王五","2")
就是这种平铺的关系。然后再使用group by。
Deployment("1")
Employee("张三","1")
Employee("李四","1")
Deployment("2")
Employee("王五","2")
现在有一个函数可以直接到达这种效果,那么就是groupjoin:
static void Main(string[] args)
{
List<Deployment> a = new List<Deployment>()
{
new Deployment("1"),
new Deployment("2"),
};
List<Employee> e = new List<Employee>()
{
new Employee("张三","1"),
new Employee("李四","1"),
new Employee("王五","2"),
};
var c = a.GroupJoin(e, deployment => deployment.Id,
employee => employee.DepartmentId,
(department, exployees) =>(department.Id, exployees)
);
Console.ReadKey();
}
其实可以看到这个exployees,最终运行时是一个Igroup类型,这和join还有group by不同之处。
然后这里写一下join和group by的。
static void Main(string[] args)
{
List<Deployment> a = new List<Deployment>()
{
new Deployment("1"),
new Deployment("2"),
new Deployment("3")
};
List<Employee> e = new List<Employee>()
{
new Employee("张三","1"),
new Employee("李四","1"),
new Employee("王五","2"),
};
var f = from a1 in a
join e1 in e on a1.Id equals e1.DepartmentId into temps
from tt in temps.DefaultIfEmpty()
select (a1.Id, (a1.Id, tt));
var d = from f1 in f
group f1 by f1.Id;
Console.ReadKey();
}
这里要说明的就是groupjoin 是一个左连接。
上面用的是linq的查询表达式,因为join 方法其实是inner join。
对于表达式来说其实最终还是生成方法而已,IL 来说没有表达式这回事,可以理解为c#的语法糖。
看下如果上述的linq用方法来怎么写吧。
其实还是用的是groupjoin:
这里可以看出,其实上述这个linq表达式就是groupjoin 然后用selectmany拆开,然后在用group再聚合。。
如果你只需要获取每个部门下的员工,用groupjoin 效率是更高的。
当然这里主要是说明linq 的表达式最终都是方法,语法糖而已。表达式是为了让我们更加清晰的去描述。
如果写一些复杂的linq,那么最好去用linq表达式。通过如果是groupjoin 就能实现的,就没必要去使用,这可能效率更低。
这里提到了selectmany,那么就讲述一下selectmany。 group by 是聚合,那么selectmany就是拆分。
public class BasketballTeam
{
public string Name { get; set; }
public string[] TeamMember;
}
static void Main(string[] args)
{
List<BasketballTeam> list = new List<BasketballTeam>();
BasketballTeam basketballTeam = new BasketballTeam();
basketballTeam.Name = "无敌球队";
basketballTeam.TeamMember = new string[] { "张三", "李四", "王五" };
BasketballTeam basketballTeam1 = new BasketballTeam();
basketballTeam1.Name = "小新球团";
basketballTeam1.TeamMember = new string[] { "张嘛子", "李老帽", "王七三" };
list.Add(basketballTeam);
list.Add(basketballTeam1);
var a = list.SelectMany(u=> u.TeamMember, (u,s)=> u.Name + " " + s);
foreach (var item in a)
{
Console.WriteLine(item);
}
Console.ReadKey();
}
selectMany 就是用来处理集合的集合的,是一个展开过程。
上面代码是遍历一组球队的全部成员。
有兴趣可以去看下代码,其实都能猜到,就是两个foreach 循环的封装。
下面介绍一下c# 匿名类型对linq使用:
如果临时使用,那么可以使用匿名类型。
匿名类型其实不是说真的匿名,其实是编译的时候会生成一个类。
如果两者完全一样,那么将会生成一个类,如果只要有一点结构不同,那么会生成两个类。
因为上面一致,所以赋值。
顺序不一致,依然生成不同的类。
不过现在元组替代了匿名类型,因为元组在方法外部也可以使用。
然后值得注意的是匿名类型是引用类型,而元组是值类型,如果经常拷贝且内存超过128位就用匿名,否则就是元组。
其实不用纠结元组和匿名,就是元组的功能比匿名强,但是因为元组是值类型,如果经常赋值拷贝的话那么肯定是引用类型更好,所以匿名类型这时候才是用武之地。
结
下一节查询表达式,也是linq的终章。后面一张就是异步(并行和并发),这个考虑在汇编系列之后,因为介绍task实现原理,需要用到il,用到il就需要一些汇编知识。
重学c#系列——linq(3) [二十九]的更多相关文章
- Web 开发人员和设计师必读文章推荐【系列二十九】
<Web 前端开发精华文章推荐>2014年第8期(总第29期)和大家见面了.梦想天空博客关注 前端开发 技术,分享各类能够提升网站用户体验的优秀 jQuery 插件,展示前沿的 HTML5 ...
- 《手把手教你》系列技巧篇(二十九)-java+ selenium自动化测试- Actions的相关操作上篇(详解教程)
1.简介 有些测试场景或者事件,Selenium根本就没有直接提供方法去操作,而且也不可能把各种测试场景都全面覆盖提供方法去操作.比如:就像鼠标悬停,一般测试场景鼠标悬停分两种常见,一种是鼠标悬停在某 ...
- 重学c#系列——字典(十一)
前言 重学c#系列继续更新,简单看一下字典的源码. 看源码主要是解释一下江湖中的两个传言: 字典foreach 顺序是字典添加的顺序 字典删除元素后,字典顺序将会改变 正文 那么就从实例化开始看起,这 ...
- FreeSql (二十九)Lambda 表达式
FreeSql 支持功能丰富的表达式函数解析,方便程序员在不了解数据库函数的情况下编写代码.这是 FreeSql 非常特色的功能之一,深入细化函数解析尽量做到满意,所支持的类型基本都可以使用对应的表达 ...
- Bootstrap <基础二十九>面板(Panels)
Bootstrap 面板(Panels).面板组件用于把 DOM 组件插入到一个盒子中.创建一个基本的面板,只需要向 <div> 元素添加 class .panel 和 class .pa ...
- 【黑金原创教程】【FPGA那些事儿-驱动篇I 】原创教程连载导读【连载完成,共二十九章】
前言: 无数昼夜的来回轮替以后,这本<驱动篇I>终于编辑完毕了,笔者真的感动到连鼻涕也流下来.所谓驱动就是认识硬件,还有前期建模.虽然<驱动篇I>的硬件都是我们熟悉的老友记,例 ...
- 使用Typescript重构axios(二十九)——添加baseURL
0. 系列文章 1.使用Typescript重构axios(一)--写在最前面 2.使用Typescript重构axios(二)--项目起手,跑通流程 3.使用Typescript重构axios(三) ...
- 剑指Offer(二十九):最小的K个数
剑指Offer(二十九):最小的K个数 搜索微信公众号:'AI-ming3526'或者'计算机视觉这件小事' 获取更多算法.机器学习干货 csdn:https://blog.csdn.net/baid ...
- WCF技术剖析之二十九:换种不同的方式调用WCF服务[提供源代码下载]
原文:WCF技术剖析之二十九:换种不同的方式调用WCF服务[提供源代码下载] 我们有两种典型的WCF调用方式:通过SvcUtil.exe(或者添加Web引用)导入发布的服务元数据生成服务代理相关的代码 ...
- VMwarevSphere 服务器虚拟化之二十九 桌面虚拟化之安装View副本服务器
VMwarevSphere 服务器虚拟化之二十九 桌面虚拟化之安装View副本服务器 VMware View中高可用性可是一个必须要考虑的问题.在整个虚拟桌面环境中View Connection S ...
随机推荐
- 3_Nginx
一. Nginx介绍 1.1 引言 为什么要学Nginx 问题1: 客户端到底要将请求发送给哪台服务器 问题2: 如果所有客户端的请求都发送给了服务器1 问题2: 客户端发送的请求可能是申请动态资源的 ...
- 【Azure 应用服务】App Service频繁出现 Microsoft.WindowsAzure.Diagnostics.DiagnosticMonitorTraceListener 异常分析
问题描述 在使用App Service的过程中,发现应用频繁出现503错误,通过Kudu站点获取到Logfiles. 在 Eventlog.xml 文件中,发现大量的 Microsoft.Window ...
- 聊聊SQL注入
SQL注入问题 概述: 首先SQL注入是一个非常危险的操作,很可能被一些不怀好意的人钻空导致我们系统出现异常等状况,比如数据库遭到破坏或被入侵. 原因:使用JDBC的Statement语句添加SQL语 ...
- KNN算法介绍及源码实现
一.KNN算法介绍 邻近算法,或者说K最邻近(KNN,K-NearestNeighbor)分类算法是数据挖掘分类技术中最简单的方法之一.所谓K最近邻,就是K个最近的邻居的意思,说的是每个样本都可以用它 ...
- 220726 T3 最优化问题 (树状数组)
题目描述 在同学们的努力下, 高匀感受到了 alb 的快乐. 高勺意犹未尽,找来了一个长度为 nn 的序列 a_1,a_2,-.,a_na1,a2,-.,an . 她想要删除这个序列中的 kk ...
- 个人音乐博客 h5、css和js等
浅说一下吧 这个小项目由h5和css还有js和jq写的 主题内容为个人音乐 博客等 首页一级导航栏 以及侧边栏 整合部分图标(侧边栏未添加收起操作 时间原因 会的朋友们可以自行添加一个动画就可以 在m ...
- 动词时态=>1.动作的时间和状态
时态 什么是时态? 英语的时态,是由动作的时间 + 动作的状态:这俩一起构成了时态 动词的时间和状态在一起,合称时态 理论上的十六种时态 先将 时间和状态的概念搞清楚,再具体讨论,用什么词,去构成时态 ...
- Linux进程间通信(一)
进程间通信 概念:进程是一个独立的资源分配单位,不同进程之间有关联,不能在一个进程中直接访问另一个进程的资源. 进程和进程之间的资源是相互独立的,一个进程不能直接访问另外一个进程的资源,但是进程和进程 ...
- 这次彻底读透 Redis
1. Redis 管道 我们通常使用 Redis 的方式是,发送命令,命令排队,Redis 执行,然后返回结果,这个过程称为Round trip time(简称RTT, 往返时间).但是如果有多条命令 ...
- visual studio插件开发-Menu
工欲善其事,必先利其器,作为程序员我们很大部分时间在和ide打交道,好的插件可以大大提高我们的编程效率,我开发过几个vs插件来解决一键生成dbmodels,快速部署到服务器,总结下来最关键的还是对于M ...