.NET中的泛型委托
.Net中有一个内置的委托 Func
它总共有以下5种形式
1. Func<TResult>
2. Func<T,TResult>
3. Func<T1,T2,TResult>
4. Func<T1,T2,T3,TResult>
5. Func<T1,T2,T3,T4,TResult>
上面5种形式,第1种 Func<TResult> 没有参数,只有返回值
我们来看一个例子
private delegate string MyDelegate(); public static string OutputDelegate()
{
return "Test delegate";
} public static void Main(string[] args)
{
MyDelegate testFunc = OutputDelegate; Console.WriteLine(testFunc()); // 输出 Test delegate }
在这个例子中,我们自己定义了一个委托MyDelegate. 但是 上面我们也说了,.Net中有自己内置的委托,或者说是.Net中默认自带的委托, 我们来看看如何使用.Net自带的委托 Func<TResult>
public static string OutputDelegate()
{
return "Test delegate";
} public static void Main(string[] args)
{
Func<string> testFunc = OutputDelegate; Console.WriteLine(testFunc()); // 输出 Test delegate }
在上面,我们使用了.net自带的委托的第一种类型Func<TResult>, 这个委托没有参数传入,返回值类型为TResult. 由于上面OutputDelegate方法中返回string类型,所以对应委托就是Func<string>
如果有参数呢,我们就需要使用其他4种形式了。 我们来看一看第2种形式委托Func<T,TResult>的使用
public static string OutputDelegate(string testStr)
{
return testStr + testStr;
} public static void Main(string[] args)
{
Func<string,string> testFunc = OutputDelegate; string strResult = testFunc("test"); Console.WriteLine(strResult); // 输出 test test }
上面的.Net内置委托Func也支持Lambda的形式调用, 我们来看一个例子
Func<string,string> myDelegate = x => x + x; Console.WriteLine(myDelegate("test")); // 输出 testtest
我们现在属性了这个内置委托Func,那么它一般在哪里使用呢,或者说在什么场景下使用比较多呢
事实上,在编程过程中,它使用最多的场景是作为函数的参数,比如下面这样
string GetResult(Func<string,string>)
在这个方法中,参数是一个代理,我们通常会通过传入一个Lambda表达式,这个代理输入参数是string类型,而返回值也是string类型,所以你看GetResult函数的返回类型就是string类型
我们知道,Linq查询操作中有一些聚合函数(count,max,min,sum,average,aggregate,longcount),我们来看看其中count的原型
public static int Count<TSource>(this IEnumerable<TSource> source);
public static int Count<TSource>(this IEnumerable<TSource> source, Func<TSource,bool> predicate);
我们从上面看到,Linq中的Count聚合函数有两个原型,第一个就是单独的计算序列中的元素的数量总和。 而第2个原型有一个委托函数,用来筛选满足一定条件的元素,然后再计算满足条件的数量和. 我们来看个实现它们的例子
private void GetCount()
{
int[] intList = new int[] {,,,,,,}; var countResult = intList.Count() //输出 7
var countResult2 = intList.Count(x=>x > ) //输出 2 }
我们再来看看其中的sum函数的一个原型
public static int Sum<TSource>(this IEnumerable<TSource> source, Func<TSource,int> selector);
看完这些,有没有啥感觉,有没有发现自己也可以写一个Linq的聚合函数,使用类似它默认自带的聚合函数同样的方法,使用扩展方法,委托作为参数,我们自己来试试, 取名为MyLinqFunc
public static TSource MyLinqFunc(this IEnumerable<TSource> source, Func<TSource, bool> funcCondition)
{
foreach(TSource item in source)
{ if(funcCondition(item))
{
return (item);
} } throw new Exception("没有满足条件的元素"); }
现在我们来写一个例子,可以使用这个自己定义的Linq的聚合函数
class myClass
{
static void Main(string[] args)
{
List<int> intList = new List<int>(){,,,,,,,};
int result = intList.MyLinqFunc(t=>t < ); //输出 1 这里使用上面我们自己定义的Linq聚合函数 MyLinqFunc
Console.WriteLine(result); } }
看了上面的分析后,我们再去看msdn上关于Enumerable.Select<TSource,TResult>方法,就更容易理解,可以自己去看https://msdn.microsoft.com/zh-cn/library/bb548891(v=vs.110).aspx https://msdn.microsoft.com/zh-cn/library/bb534869(v=vs.110).aspx
上面我们讨论了.Net中的泛型委托,我们知道,还有一个概念,就是委托表达式,这样一种形式Expression<Func<TSource,TResult>>. 那么这两者到底有什么区别呢。这是一个很容易让人摸不清头脑的问题
我们来看看
Func<TSource,TResult> 是委托
Expression<Func<TSource,TResult>>是表达式
Expression要编译后(调用Expression的Compile方法)才能变成delegate, 才能运行。 举个例子如下
Expression<Func<int,bool>> ex = x => x > ;
Func<int,bool> myFunc = ex.Compile();
现在我们就可以调用myFunc:
myFunc(5) //返回false
myFunc(12) //返回true
而表达式是不可以直接调用的
但是在Linq to Entity的Where查询条件中,就应该使用Expression,而不是使用委托。 比如如下是错误的
Func<tbUser,bool> existFunc = u => u.userID == id && u.userName == name; _tbUserRepository.Entities.Where(existFunc);
上面这里,where里面的参数是一个委托,这个是错误的。实际上,where里面需要的参数是Expression, 所以正确的方式应该如下
Expression<Func<tbUser,bool>> existFunc = u => u.userID == id && u.userName == name; _tbUserRepository.Entities.Where(existFunc);
.NET中的泛型委托的更多相关文章
- List<string>中的泛型委托
我们先看List<T>.Sort().其定义是:public void Sort( Comparison<T> comparison ) 其要求传入的参数是Comparison ...
- 委托学习笔记后续:泛型委托及委托中所涉及到匿名方法、Lambda表达式
引言: 最初学习c#时,感觉委托.事件这块很难,其中在学习的过程中还写了一篇学习笔记:委托.事件学习笔记.今天重新温故委托.事件,并且把最近学习到和委托相关的匿名方法.Lambda表达式及泛型委托记录 ...
- .NET中的Action及Func泛型委托
委托,在C#编程中占有极其重要的地位,委托可以将函数封装到委托对象中,并且多个委托可以合并为一个委托,委托对象则可以像普通对象一样被存储.传递,之后在任何时刻进行调用,因此,C#中函数回调机制的实现基 ...
- 使用.NET中的Action及Func泛型委托
委托,在C#编程中占有极其重要的地位,委托可以将函数封装到委托对象中,并且多个委托可以合并为一个委托,委托对象则可以像普通对象一样被存储.传递,之后在任何时刻进行调用,因此,C#中函数回调 ...
- C#中Predicate<T>与Func<T, bool>泛型委托的用法实例
本文以实例形式分析了C#中Predicate<T>与Func<T, bool>泛型委托的用法,分享给大家供大家参考之用.具体如下: 先来看看下面的例子: 1 2 3 4 5 6 ...
- 泛型委托及委托中所涉及到匿名方法、Lambda表达式
泛型委托及委托中所涉及到匿名方法.Lambda表达式 引言: 最初学习c#时,感觉委托.事件这块很难,其中在学习的过程中还写了一篇学习笔记:委托.事件学习笔记.今天重新温故委托.事件,并且把最近学习到 ...
- c#打包文件解压缩 C#中使用委托、接口、匿名方法、泛型委托实现加减乘除算法 一个简单例子理解C#的协变和逆变 对于过长字符串的大小比对
首先要引用一下类库:using Ionic.Zip;这个类库可以到网上下载. 下面对类库使用的封装方法: 得到指定的输入流的ZIP压缩流对象 /// <summary> /// 得到指定的 ...
- C#中使用委托、接口、匿名方法、泛型委托实现加减乘除算法
使用C#实现加减乘除算法经常被用作新手练习.本篇来分别体验通过委托.接口.匿名方法.泛型委托来实现. 使用委托实现 加减乘除拥有相同的参数个数.类型和返回类型,首先想到了使用委托实现. //创建一个委 ...
- 用五分钟重温委托,匿名方法,Lambda,泛型委托,表达式树
这些对老一代的程序员都是老生常谈的东西,没什么新意,对新生代的程序员却充满着魅力.曾经新生代,好多都经过漫长的学习,理解,实践才能掌握委托,表达式树这些应用.今天我尝试用简单的方法叙述一下,让大家在五 ...
随机推荐
- 使用deepfashion实现自己的第一个分类网络
这个过程主要分为三个步骤: 数据预处理 数据处理就是把数据按照一定的格式写出来,以便网路自己去读取数据 1准备原始数据 我的cloth数据一共是四个类别,每个类别有衣服47张,一用是188张图片,这些 ...
- javascript箭头函数把函数给简写了[0403]
箭头函数把函数给简写了[0403] 我不是很喜欢箭头函数,总觉得它让原本就不那么严谨的js更加不严谨了,所以有时候看js程序也是一件很头痛的事情,不过在ES6中加入了这么一个新的方法,已 ...
- float元素的父元素自适应高度
当在对象内的盒子使用了float后,导致对象本身不能被撑开自适应高度,这个是由于浮动产生原因. 如何解决父div对象自适应高度,方法有三种. 1.对父元素设置固定高度 2.使用clear清除浮动 3. ...
- Elipse 快捷键
1. eclipse里面如何快速收缩当前类文件里面的所有方法和注释收缩:ctrl+shift+/展开:ctrl+shift+*注意:这个/和*要是数字键盘上的/和*. 2. shift+enter ...
- Nginx Rewrite语法详解
重写中用到的指令 if (条件) {} 设定条件,再进行重写 set #设置变量 return #返回状态码 return 403; break #跳出rewrite rewrite #重写 I ...
- Spring Cloud之Swagger集群搭建
在微服务中,Swagger是每个服务 比如会员服务,订单服务,支付服务 进行继承. 如何将整个微服务中的Swagger进行合成,同一台服务器上. 使用Zuul+Swagger实现管理整个微服务API文 ...
- docker网络模型
docker run -it --rm --net none --name test centos:newer /bin/bash --net none的作用是创建一个封闭的容器,容器只有lo接口,只 ...
- 随意谈谈tcp
tcp作为四层中可靠到传输协议,为上层协议提供了字节流的可靠到传输,之所以能做到可靠主要因为以下几点: 1.流与分段:流即字节流,计算机处理程序时一般以字节为单位,如果上层协议接收到到是字节流并且跟发 ...
- jsp操作xml
<?xml version="1.0" encoding="UTF-8"?> <!-- 说明是xml文件,文件的版本和字符编码 --> ...
- HDU 4034 Graph:反向floyd
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4034 题意: 有一个有向图,n个节点.给出两两节点之间的最短路长度,问你原图至少有多少条边. 如果无解 ...