百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

C# 基础知识系列- 6 Lambda表达式和Linq简单介绍

bigegpt 2024-08-18 14:13 3 浏览

前言

C#的lambda和Linq可以说是一大亮点,C#的Lambda无处不在,Linq在数据查询上也有着举足轻重的地位。

那么什么是Linq呢,Linq是 Language Intergrated Query(语言集成查询)的缩写,可以对本地对象集合或者远程数据源进行结构化的查询操作。

那什么又是Lambda呢?嗯,简单来讲就是匿名函数,我们不声明方法名,只写一个方法体,这个方法体就是lambda表达式



lambda表达式


如何写一个lambda表达式

首先,在写lambda表达式之前,需要先了解 两个特殊的类型:Func和Action。

这是两个委托,这里先不急着了解什么是委托,可以把它们当做一种名称规范就行,它们都可以表示一个方法。不同的是其中Func表示一个有返回值的方法,Action表示一个没有返回值的方法。C#对这两个的定义如下:

public delegate TResult Func<out TResult>();//注意这里的out 表示这个泛型是返回值的类型泛型
public delegate void Action();

其中Func和Action各有16个变种:

// 注意 in 关键字,表示泛型是参数的类型约束
public delegate TResult Func<in T,out TResult>(T arg);
public delegate TResult Func<in T1,in T2,out TResult>(T1 arg1, T2 arg2);
……
public delegate TResult Func<in T1,in T2,in T3,in T4,in T5,in T6,in T7,in T8,in T9,in T10,in T11,in T12,in T13,in T14,in T15,in T16,out TResult>(T1 arg1, T2 arg2, T3 arg3, T4 arg4, T5 arg5, T6 arg6, T7 arg7, T8 arg8, T9 arg9, T10 arg10, T11 arg11, T12 arg12, T13 arg13, T14 arg14, T15 arg15, T16 arg16);
//
//
?
public delegate void Action<in T>(T obj);
public delegate void Action<in T1,in T2>(T1 arg1, T2 arg2);
……
public delegate void Action<in T1,in T2,in T3,in T4,in T5,in T6,in T7,in T8,in T9,in T10,in T11,in T12,in T13,in T14,in T15,in T16>(T1 arg1, T2 arg2, T3 arg3, T4 arg4, T5 arg5, T6 arg6, T7 arg7, T8 arg8, T9 arg9, T10 arg10, T11 arg11, T12 arg12, T13 arg13, T14 arg14, T15 arg15, T16 arg16);


依次表示一个参数、两个参数、……十六个参数 的方法。当然,你还可以写更多的参数,但是如果一个方法的参数超过10个,为什么不用类封装起来呢?即使不封装,一个方法十几个参数,你确定不会被你的领导嫌弃吗。


言归正传,介绍完了Func和Action的定义,那么如果使用呢?

public void Demo1()
{
    // 一个没有返回值,也没有参数的方法
}
?
Action act1 = Demo;// 直接给一个方法名
?
public void Demo2(string name)
{
    //有一个参数,但没有返回值的方法
}
?
Action<string> act2 = Demo2;
?
public String Demo3()
{
    // 有返回值,但没有参数的方法
}
Func<string> func1 = Demo3;
?
public int Demo4(double data)
{
    // 返回值是int,有一个参数是double的方法
}
?
Func<double,int> func2 = Demo4;
?
?

以上是通过方法名获取Func和Action的方法,下面介绍一下通过Lambda表达式的方式创建Func和Action:

Action act1 = ()=> // lambda 的标志性 声明方式 =>
{
    // 这是一个没有返回值,也没有参数的 lambda表达式
};
Action<int> act2 = (age) => 
{
    // 这是一个 参数为int ,没有返回值的 lambda表达式
};
//=========================================
Func<string> func1 = () => ""; // 这是一个返回了空字符串的lambda表达式,注意这种写法
Func<string> func2 = () =>
{
    return ""; //与上一个等价
}
?
Func<int,string> func3 = (age) =>
{
    return "我的年纪是:"+age;// 一个参数是int,返回类型是string的lambda表达式
}

在lambda表达式中,当使用的是有返回值的方法体时,如果方法体是个简单的计算式或者说可以在一行内写完(或被编译器认为是一行)的话,可以省略 {、} 和return,直接用 => 标记。

比如说以下内容:

Func<int,int,int> cal_area = (width, height) => width * height;// 计算面积


使用Lambda 表达式

现在我们手里有一大堆的Action和Func,我们该怎么用呢?

有以下两种常见的用法:

  1. 把它当做方法来用:// 上接上文代码
    act1();// 执行 act1 代表的方法或lambda表达式
    act2(10); //执行act2 的lambda表达式
    ?
    string str1 = func1();
    string str2 = func3(10);
    int area = cal_area(29,39);
  2. 调用Invoke方法:act1.Invoke();
    act2.Invoke(10);
    ?
    area = cal_area.Invoke(33,63);看过反射篇的应该对Invoke有一定印象,这个与MethodInfo里的Invoke类似,但是比其更加简单。



Linq 是什么

正如前言所述,Linq是一种对集合、数据源的集成式查询方式,它是对IEnumerable<T>的扩展方法集,所以想要使用Linq的话,需要引用两个命名空间 System.Linq和System.Linq.Expressions。


Linq有两种使用方式,一种是通过方法链的方式调用,一种是类似SQL语句的方式进行数据查询。方法链是基础,类SQL方式是语法糖。下面简单介绍一下两种方式的使用,不过首先先假设我们有一个数据很多的集合:

IEnumerable<int> scores = new List<int>();//假设存放了某班50个人的语文成绩


使用方法链查询


  1. 获取分数大于60的所有分数:IEnumerable<int> result1 = scores.Where(t => t > 60);
  2. 获取分数大于等于60的数量:int count = scores.Count(t => t >= 60);
  3. 统计分数总和int sum = scores.Sum();
  4. 获取所有分数个位上的数字:IEnumerable<int> result2 = scores.Select(t => t % 10);


使用类SQL形式查询

查询所有大于等于60的分数:

IEnumerable<int> result3 = from score in scores
                where score >= 60
                select score;

简单介绍一下,类SQL形式有一个统一的格式写法,关键字from、in、select缺一不可:

from 临时变量名 in 数据源

select 结果类型

where 是条件过滤,如果查询全部,可以忽略。

这种方式之所以被我称为是类SQL形式,是因为它的写法和SQL及其相似,熟悉SQL的可以很快上手。


为什么说方法链是基础呢?

因为SQL形式的查询里每一个关键字背后都有一个方法作为支撑,除了from 和in。

select 对应的Select 方法,where对应的Where方法。


需要特别注意的一点:

Linq查询是一种延迟查询,也就是说当返回类型是一个IEnumerable 的时候不会立即返回结果,必须调用ToList才能获取到实际查询结果。另外需要注意的是,ToList返回的是一个不可变List集合,这一点在集合篇中做过介绍了。


未完待续

C#里的Linq内容如此丰富,以至于一时间无法详细说明,后续还会有两到三篇关于Linq的内容,今天就先到这里了。

相关推荐

悠悠万事,吃饭为大(悠悠万事吃饭为大,什么意思)

新媒体编辑:杜岷赵蕾初审:程秀娟审核:汤小俊审签:周星...

高铁扒门事件升级版!婚宴上‘冲喜’老人团:我们抢的是社会资源

凌晨两点改方案时,突然收到婚庆团队发来的视频——胶东某酒店宴会厅,三个穿大红棉袄的中年妇女跟敢死队似的往前冲,眼瞅着就要扑到新娘的高额钻石项链上。要不是门口小伙及时阻拦,这婚礼造型团队熬了三个月的方案...

微服务架构实战:商家管理后台与sso设计,SSO客户端设计

SSO客户端设计下面通过模块merchant-security对SSO客户端安全认证部分的实现进行封装,以便各个接入SSO的客户端应用进行引用。安全认证的项目管理配置SSO客户端安全认证的项目管理使...

还在为 Spring Boot 配置类加载机制困惑?一文为你彻底解惑

在当今微服务架构盛行、项目复杂度不断攀升的开发环境下,SpringBoot作为Java后端开发的主流框架,无疑是我们手中的得力武器。然而,当我们在享受其自动配置带来的便捷时,是否曾被配置类加载...

Seata源码—6.Seata AT模式的数据源代理二

大纲1.Seata的Resource资源接口源码2.Seata数据源连接池代理的实现源码3.Client向Server发起注册RM的源码4.Client向Server注册RM时的交互源码5.数据源连接...

30分钟了解K8S(30分钟了解微积分)

微服务演进方向o面向分布式设计(Distribution):容器、微服务、API驱动的开发;o面向配置设计(Configuration):一个镜像,多个环境配置;o面向韧性设计(Resista...

SpringBoot条件化配置(@Conditional)全面解析与实战指南

一、条件化配置基础概念1.1什么是条件化配置条件化配置是Spring框架提供的一种基于特定条件来决定是否注册Bean或加载配置的机制。在SpringBoot中,这一机制通过@Conditional...

一招解决所有依赖冲突(克服依赖)

背景介绍最近遇到了这样一个问题,我们有一个jar包common-tool,作为基础工具包,被各个项目在引用。突然某一天发现日志很多报错。一看是NoSuchMethodError,意思是Dis...

你读过Mybatis的源码?说说它用到了几种设计模式

学习设计模式时,很多人都有类似的困扰——明明概念背得滚瓜烂熟,一到写代码就完全想不起来怎么用。就像学了一堆游泳技巧,却从没下过水实践,很难真正掌握。其实理解一个知识点,就像看立体模型,单角度观察总...

golang对接阿里云私有Bucket上传图片、授权访问图片

1、为什么要设置私有bucket公共读写:互联网上任何用户都可以对该Bucket内的文件进行访问,并且向该Bucket写入数据。这有可能造成您数据的外泄以及费用激增,若被人恶意写入违法信息还可...

spring中的资源的加载(spring加载原理)

最近在网上看到有人问@ContextConfiguration("classpath:/bean.xml")中除了classpath这种还有其他的写法么,看他的意思是想从本地文件...

Android资源使用(android资源文件)

Android资源管理机制在Android的开发中,需要使用到各式各样的资源,这些资源往往是一些静态资源,比如位图,颜色,布局定义,用户界面使用到的字符串,动画等。这些资源统统放在项目的res/独立子...

如何深度理解mybatis?(如何深度理解康乐服务质量管理的5个维度)

深度自定义mybatis回顾mybatis的操作的核心步骤编写核心类SqlSessionFacotryBuild进行解析配置文件深度分析解析SqlSessionFacotryBuild干的核心工作编写...

@Autowired与@Resource原理知识点详解

springIOCAOP的不多做赘述了,说下IOC:SpringIOC解决的是对象管理和对象依赖的问题,IOC容器可以理解为一个对象工厂,我们都把该对象交给工厂,工厂管理这些对象的创建以及依赖关系...

java的redis连接工具篇(java redis client)

在Java里,有不少用于连接Redis的工具,下面为你介绍一些主流的工具及其特点:JedisJedis是Redis官方推荐的Java连接工具,它提供了全面的Redis命令支持,且...