Func、Action 的区别于说明

1、Func

Func是一个.Net内置的委托。设计模式

Func<Result>,Func<T1,Result>是一个.Net内置的泛型委托。函数

  • Func<TResult>
  • Func<T,TResult>
  • Func<T1,T2,TResult>
  • Func<T1,T2,T3,TResult>
  • Func<T1,T2,T3,T4,TResult>

它有5种形式,只是参数个数不一样;第一个是无参数,可是有返回值;this

下面是一个简单的普通委托来传方法的示例。spa

        private delegate string Say();
        public static string SayHello()
        {
            return "Hello";
        }

        static void Main(string[] args)
        {
            Say say = SayHello;
            Console.WriteLine(say());
            Console.ReadKey();
        }

因此,在有时候,咱们不知道一个接口同时要作什么操做的时候,我能够给它留一个委托。设计

为了更方便,.Net直接默认有了委托。咱们再来试试.Net默认带的委托。blog

        public static string SayHello()
        {
            return "Hello";
        }

        static void Main(string[] args)
        {
            Func<string> say = SayHello;
            Console.WriteLine(say());
            Console.ReadKey();
        }

若是须要参数的,还能够这样传一份。接口

        public static string SayHello(string str)
        {
            return str + str;
        }

        static void Main(string[] args)
        {
            Func<string, string> say = SayHello;
            string str = say("abc");    
            Console.WriteLine(str);     //输出abcabc
            Console.ReadKey();
        }

2、Action

Action<T>的用法与Func几乎同样,调用方法也相似。string

  • Action
  • Action<T>
  • Action<T1,T2>
  • Action<T1,T2,T3>
  • Action<T1,T2,T3,T4>
        private delegate string Say();
        public static void SayHello(string str)
        {
            Console.WriteLine(str);
        }

        static void Main(string[] args)
        {
            Action<string> say = SayHello;
            say("abc");
            Console.ReadKey();
        }

3、Func与Action的区别

Func与Action做用几乎同样。只是it

  • Func<Result>有返回类型;
  • Action<T>只有参数类型,不能传返回类型。因此Action<T>的委托函数都是没有返回值的。

4、Func与Action都支持Lambda的形式调用

仍是以一个输入后,返回重复一次的值做为示例。io

Func<string, string> say = m => m + m;
Console.WriteLine(say("abc"));    //输出abcabc

5、最多见到Func的地方

一般咱们最多见到Func是在方法的参数里以下面这样:

string XXX(Func<string, string>)

我们来看看Linq里面的其中一个Sum:

public static int Sum<TSource>(this IEnumerable<TSource> source, Func<TSource, int> selector);

里面看到两点:
一、扩展方法,与这篇文章无关(扩展的是IEnumerable<TSource>,主要是为了可以实现IEnumerable<TSource>接口的集合.出函数)。
二、Func<TSource, int> selector这个参数。
尝试写一个Linq的First函数吧,命名为First2。Linq源代码里有不少异常状况处理,好多设计模式,惋惜我不懂,只提取简单逻辑了。

namespace ConsoleApplication2
{
    static class Extend
    {
        public static TSource First2<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate)
        {
            //.Net自己的源代码好多异常状况处理,好多设计模式,我也不懂,只提取逻辑
            foreach (TSource item in source)
            {
                if (predicate(item))
                {
                    return (item);
                }
            }
            throw new Exception("不存在知足条件的第一个元素!");
        }
    }

    class Program
    {
        static void Main(string[] args)
        {
            List<int> ListInt = new List<int>(){ 1, 2, 3, 4, 5 };
            int k = ListInt.First2(m => m > 4);     //输出5
            Console.WriteLine(k);

            Console.ReadKey();
        }
    }
}
相关文章
相关标签/搜索