目录

写在前面

系列文章

带有标准查询运算符的Lambda

Lambda中类型推断

Lambda表达式中变量作用域

异步Lambda

总结

写在前面

上篇文章介绍了Lambda的基本概念以及匿名方法,本篇继续介绍Lambda的一些内容,既然学了,就要总结的全面一点。

系列文章

Linq之Lambda表达式初步认识

带有标准查询运算符的Lambda

什么事标准查询运算符?

“标准查询运算符”是组成语言集成查询 (LINQ) 模式的方法。 大多数这些方法都在序列上运行,其中的序列是一个对象,其类型实现了 IEnumerable<T> 接口或 IQueryable<T> 接口。 标准查询运算符提供了包括筛选、投影、聚合、排序等功能在内的查询功能。
共有两组 LINQ 标准查询运算符,一组在类型为 IEnumerable<T> 的对象上运行,另一组在类型为 IQueryable<T> 的对象上运行。 构成每组运算符的方法分别是 Enumerable 和 Queryable 类的静态成员。 这些方法被定义为作为方法运行目标的类型的“扩展方法”。 这意味着可以使用静态方法语法或实例方法语法来调用它们。
此外,许多标准查询运算符方法运行所针对的类型不是基于 IEnumerable<T> 或 IQueryable<T> 的类型。 Enumerable 类型定义两个此类方法,这些方法都在类型为 IEnumerable 的对象上运行。 利用这些方法(Cast<TResult>(IEnumerable) 和 OfType<TResult>(IEnumerable)),您将能够在 LINQ 模式中查询非参数化或非泛型集合。 这些方法通过创建一个强类型的对象集合来实现这一点。 Queryable 类定义两个类似的方法(Cast<TResult>(IQueryable) 和 OfType<TResult>(IQueryable)),这些方法在类型为 Queryable 的对象上运行。
各个标准查询运算符在执行时间上有所不同,具体情况取决于它们是返回单一值还是值序列。 返回单一值的方法(例如 Average 和 Sum)会立即执行。 返回序列的方法会延迟查询执行,并返回一个可枚举的对象。
对于在内存中集合上运行的方法(即扩展 IEnumerable<T> 的那些方法),返回的可枚举对象将捕获传递到方法的参数。 在枚举该对象时,将使用查询运算符的逻辑,并返回查询结果。
与之相反,扩展 IQueryable<T> 的方法不会实现任何查询行为,但会生成一个表示要执行的查询的表达式树。 查询处理由源 IQueryable<T> 对象处理。
可以在一个查询中将对查询方法的调用链接在一起,这就使得查询的复杂性可能会变得不确定。

来自:http://msdn.microsoft.com/zh-cn/library/bb397896.aspx

多数标准查询运算符都有输入参数,其类型是17个.net泛型委托Func<T,TResult>中的一种。

比如:

         //
// 摘要:
// 返回一个数字,表示在指定的序列中满足条件的元素数量。
//
// 参数:
// source:
// 包含要测试和计数的元素的序列。
//
// predicate:
// 用于测试每个元素是否满足条件的函数。
//
// 类型参数:
// TSource:
// source 中的元素的类型。
//
// 返回结果:
// 一个数字,表示序列中满足谓词函数条件的元素数量。
//
// 异常:
// System.ArgumentNullException:
// source 或 predicate 为 null。
//
// System.OverflowException:
// source 中的元素数量大于 System.Int32.MaxValue。
public static int Count<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate);

从上面的代码中可以看出Count方法扩展自IEnumerable<TSource>,并且有输入参数Func<TSource, bool> predicate。

一个例子

1.判断输入参数是否大于5,调用myFunc(4),如果大于5返回true,否则返回false。

2.输入一个int类型的整数,并输入一个string类型的字符串,判断拼接后的字符串是否为空,并返回bool类型的结果。

             //其中 int 是输入参数,bool 是返回值 返回值始终在最后一个类型参数中指定
Func<int, bool> myFunc = x => x > ;
bool result = myFunc();
//其中 int string是输入参数,bool 是返回值 返回值始终在最后一个类型参数中指定
//上篇文章中已经指出,在有多个输入参数的情况下,必须使用括号,输入参数以逗号隔开
Func<int, string, bool> myFunc2 = (x, y) => string.IsNullOrEmpty(x.ToString() + y);

当参数类型为 Expression<Func> 时,也可以提供 Lambda 表达式,例如在 System.Linq.Queryable 内定义的标准查询运算符中, 如果指定 Expression<Func> 参数,lambda 将编译为表达式目录树。

一个例子

此处列举一个标准查询运算符,Count 方法:

            //一个整数数组
int[] numbers = { , , , , , , , , , };
//计算整数数组中奇数的个数
int oddNumbers = numbers.Count(n => n % == );
Console.WriteLine(oddNumbers);

在这里你会发现,n直接使用n%2,编译器将推断n的类型为整型。

从上面的定义及例子,发现标准查询运算符,有这样的特点:1,查询什么(集合或者数组,集合和数组有什么特点?要么实现了IEnumerable<T> 接口,要么IQueryable<T> 接口)。

Lambda中类型推断

在编写 lambda 时,通常不必为输入参数指定类型,因为编译器可以根据 lambda 主体、参数的委托类型以及 C# 语言规范中描述的其他因素来推断类型。 对于大多数标准查询运算符,第一个输入是源序列中的元素类型。 因此,如果要查询 IEnumerable<Customer>,则输入变量将被推断为 Customer 对象,这意味着你可以访问其方法和属性:

 customers.Where(c => c.City == "London");

Lambda的一般规则:

Lambda 包含的参数数量必须与委托类型包含的参数数量相同。
Lambda 中的每个输入参数必须都能够隐式转换为其对应的委托参数。
Lambda 的返回值(如果有)必须能够隐式转换为委托的返回类型。

Lambda表达式中变量作用域

lambda表达式中变量的作用域在定义 lambda 函数的方法内或包含 lambda 表达式的类型内,lambda 可以引用范围内的外部变量。 以这种方式捕获的变量将进行存储以备在 lambda 表达式中使用,即使在其他情况下,这些变量将超出范围并进行垃圾回收。 必须明确地分配外部变量,然后才能在 lambda 表达式中使用该变量。

一个例子

 using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;
using System.Threading.Tasks; namespace Wolfy.LinqDemo
{
delegate bool D();
delegate bool D2(int i); class Program
{
static D del;
static D2 del2;
static void Main(string[] args)
{
//调用TestMethod方法
TestMethod();
// Prove that del2 still has a copy of
// local variable j from TestMethod.
//证明del2仍保留方法TestMethod中变量j的副本。
bool result = del2();
//输出true
Console.WriteLine(result);
Console.ReadKey();
}
static void TestMethod(int input)
{
int j = ;
// 使用lambda表达式初始化del.
// Note access to 2 outer variables.
// del will be invoked within this method(del将在这个方法内部执行).
del = () => { j = ; return j > input; };
// del2 will be invoked after TestMethod goes out of scope.
//del2将在TestMethod外部执行。
del2 = (x) => { return x == j; };
// Demonstrate value of j:
//展示j的值
// Output: j = 0
//输出j=0
// The delegate has not been invoked yet.
//委托仍没有执行
Console.WriteLine("j = {0}", j);
// Invoke the delegate.
//委托执行
bool boolResult = del();
// Output: j = 10 b = True
//输出j=10 b=True
Console.WriteLine("j = {0}. b = {1}", j, boolResult);
} }
}

在执行TestMethod中的以下代码时:

             // Invoke the delegate.
//委托执行
bool boolResult = del();
// Output: j = 10 b = True
//输出j=10 b=True
Console.WriteLine("j = {0}. b = {1}", j, boolResult);

调用del()修改了j的值为10,在Main方法中调用del2(10),此时仍保留方法TestMehod中j的副本。所以此时输出为:

适用于 lambda 表达式中的变量范围的规则

捕获的变量将不会被作为垃圾回收,直至引用变量的委托符合垃圾回收的条件。
在外部方法中看不到 lambda 表达式内引入的变量。
Lambda 表达式无法从封闭方法中直接捕获 ref 或 out 参数。
Lambda 表达式中的返回语句不会导致封闭方法返回。
如果跳转语句的目标在块外部,则 lambda 表达式不能包含位于 lambda 函数内部的 goto 语句、break 语句或 continue 语句。 同样,如果目标在块内部,则在 lambda 函数块外部使用跳转语句也是错误的。

异步Lambda

通过使用 async 和 await 关键字,你可以轻松创建包含异步处理的 lambda 表达式和语句。

一个例子

在winform的单击事件,异步的方式调用方法ExampleMethodAsync。

        private async void button1_Click(object sender, EventArgs e)
{
await ExampleMethodAsync();
}
async Task ExampleMethodAsync()
{
// 下面模拟一个任务返回的异步进程。
await Task.Delay();
}

如果使用异步Lambda,你可以这样写,注意在lambda前加上async关键字。

  button1.Click += async (sender, e) =>
{
await ExampleMethodAsync();
};
async Task ExampleMethodAsync()
{
await Task.Delay();
}

看上去更简洁。

总结

本篇文章介绍了lambda标准查询运算符,变量范围,类型推断,异步等概念。其他的还好理解,唯有变量范围太绕了,也只有做个记录慢慢体会了。

参考文章

http://msdn.microsoft.com/zh-cn/library/bb397896.aspx

http://msdn.microsoft.com/zh-cn/library/bb397687.aspx

Linq之Lambda进阶的更多相关文章

  1. Linq之Expression进阶

    目录 写在前面 系列文章 表达式树解析 表达式树特性 编译表达树 总结 写在前面 让我们首先简单回顾一下上篇文章介绍的内容,上篇文章介绍了表达式树的基本概念(表达式树又称为“表达式目录树”,以数据形式 ...

  2. c#封装DBHelper类 c# 图片加水印 (摘)C#生成随机数的三种方法 使用LINQ、Lambda 表达式 、委托快速比较两个集合,找出需要新增、修改、删除的对象 c# 制作正方形图片 JavaScript 事件循环及异步原理(完全指北)

    c#封装DBHelper类   public enum EffentNextType { /// <summary> /// 对其他语句无任何影响 /// </summary> ...

  3. LINQ使用Lambda表达式选择几列

    学习LINQ的Lambda的表达式,尝试从数据集合中,选择其中几列. 创建一个model: source code: namespace Insus.NET.Models { public class ...

  4. SQL、LINQ、Lambda 三种用法(转)

    SQL.LINQ.Lambda 三种用法颜色注释: SQL LinqToSql Lambda QA1. 查询Student表中的所有记录的Sname.Ssex和Class列.select sname, ...

  5. 浅谈sql 、linq、lambda 查询语句的区别

    浅谈sql .linq.lambda 查询语句的区别 LINQ的书写格式如下: from 临时变量 in 集合对象或数据库对象 where 条件表达式 [order by条件] select 临时变量 ...

  6. LInq 与lambda表达式

    LInq 与lambda表达式 LinQ是我们常用的技术之一.因为我们绕不开的要对数据进行一系列的调整,如 排序. 条件筛选.求和.分组.多表联接 等等. lambda则是我们常用的语法糖,配合lin ...

  7. 使用LINQ、Lambda 表达式 、委托快速比较两个集合,找出需要新增、修改、删除的对象

    本文需要对C#里的LINQ.Lambda 表达式 .委托有一定了解. 在工作中,经常遇到需要对比两个集合的场景,如: 页面集合数据修改,需要保存到数据库 全量同步上游数据到本系统数据库 在这些场景中, ...

  8. Linq与Lambda,神一般的工作效率

    Linq与Lambda,神一般的工作效    通过对linq和lambda的学习,越发感觉linq和lambda的重要性,他们能极大地简化程序,同时提升程序的可读性,大大提升了我们的工作效率,在公司的 ...

  9. SQL、Linq和Lambda表达式 的关系

    首先说说这三者完全是三种不同的东西,SQL是结构化查询语言(Structured Query Language)简称,这大家再熟悉不过了,下面主要介绍LINQ和Lambda表达式的基本概念以及同一查询 ...

随机推荐

  1. MCS51系列单片机实用技术部分课件

  2. esxi安装全过程及基本配置

    esxi6.0下载地址 链接: http://pan.baidu.com/s/1jIfg2yU 密码: qacv 支持检测可以参考:http://www.linuxidc.com/Linux/2012 ...

  3. hdu 2583 permutation

    permutation Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tota ...

  4. web.xml文件报错:cvc-complex-type.2.4.a: Invalid content was found starting with element 'init-param'.

    <?xml version="1.0" encoding="UTF-8"?> <web-app version="3.0" ...

  5. 一个Azure VM RDP连接问题

    由于Azure上的VM都是通过同一个镜像文件创建的,有时会需要修改SID. 在给一台VM修改SID重启后,就无法通过RDP连接到虚机了,从Azure管理界面的启动诊断界面上可以看到虚拟停在一个要求用户 ...

  6. 读高性能JavaScript编程学英语 第一章第三页第一段话

    When the browser encounters a <script> tag, as in this HTML page, there is no way of knowing w ...

  7. 【Android UI设计与开发】2.引导界面(二)使用ViewPager实现欢迎引导页面

    1.实现的效果 2.编码前的准备工作 ViewPager是Android3.0之后提供的新特性,所以要想让你的应用向下兼容就必须要android-support-v4.jar这个包的支持,这是一个来自 ...

  8. 合工大OJ 1330 种树

    Description 某校大门外长度为L的马路上有一排树,每两棵相邻的树之间的间隔都是1米.我们可以把马路看成一个数轴,马路的一端在数轴0的位置,另一端在L的位置:数轴上的每个整数点,即0,1,2, ...

  9. POJ 1269 Intersecting Lines【判断直线相交】

    题意:给两条直线,判断相交,重合或者平行 思路:判断重合可以用叉积,平行用斜率,其他情况即为相交. 求交点: 这里也用到叉积的原理.假设交点为p0(x0,y0).则有: (p1-p0)X(p2-p0) ...

  10. 用SqlParameter 给SQL传递参数

    1.数据访问层 using的用法: 01.可以using System;导命名控空间 02.using 的语法结构 using(变量类型  变量名 =new 变量类型()) { } 案例: 03.us ...