读书笔记之C# delegate

c#代理的使用主要在:需要将一个方法当做参数传递到另一个方法时。

比如启动一个线程执行任务,而这个线程要执行的方法可以通过代理传递过来。

代理包括一个方法或者多个方法的地址和C++的函数指针很相似,但它是类型安全的。

1.声明代理

delegate void IntMethodInvoker(int x);

这个代理可以引用返回值类型为空,有一个参数的所有方法,很重要的一点需要知道delegate是类型安全的。

delegate double TwoLongsOp(long first, long second);

有两个参数返回值为double的代理。

delegate string GetAString();

没有参数返回值为string的代理。

public delegate string GetAString();

还可以说明代理的访问权限。

2.使用代理

使用代理必须先要实例化一个代理

pri v a te delegate string GetAString();
static void Main()
{
int x = 40;
GetAString firstStringMethod = new GetAString(x.ToString); //实例化一个代理,并把该代理指向x.tostring();
Console.WriteLine("String is {0}", firstStringMethod()); //firstStringMethod()成为x.tostring()代表(代理)
// With firstStringMethod initialized to x.ToString(),
// the above statement is equivalent to saying
// Console.WriteLine("String is {0}", x.ToString());
}

 可以通过以下两种方法来调用已经实例化的方法

firstStringMethod();
firstStringMethod.Invoke();

当然在实例化一个代理的时候并传递方法也可以通过以下两种形式

GetAString firstStringMethod = new GetAString(x.ToString);

GetAString firstStringMethod = x.ToString;

请看下面代理实例

    class mathOperation
    {
        public static double MultiplyByTwo(double value)
        {
            return value * 2;
        }

        public static double Square(double value)
        {
            return value * value;
        }
    }

定义一个方法,该方法有两个静态函数,分别用于计算值得倍数和值的平方。

   class Program
    {
        delegate double DoubleOp(double x);  //定义代理用于代理相应的方法
        static void Main(string[] args)
        {
            DoubleOp[] operations =
            {
                mathOperation.MultiplyByTwo,
                mathOperation.Square
            }; //实例化一个代理数组,第一个数组元素代理MultiplyByTwo方法第二个数组元素代理Square方法
            for(int i =0 ; i < operations.Length;i++)
            {
                Console.WriteLine("Using operations[{0}]:", i);
                ProcessAndDisplayNumber(operations[i], 2.0);
                ProcessAndDisplayNumber(operations[i], 7.94);
                ProcessAndDisplayNumber(operations[i], 1.414);
                Console.WriteLine();
            }

        }

        static void ProcessAndDisplayNumber(DoubleOp action,double value) //代理作为参数进行传输
        {
            double result = action(value);
            Console.WriteLine("Value is {0} , result of operation is {1}", value, result);
        }
    }

二、如果不想定义代理类型而使用代理可以使用下列代理形式

Action<T> and Func<T> Delegates

action<T>指向没有返回类型的方法,可以传递16个参数,其形式如下:

 Action<in T> 指向一个参数的方法

 Action<in T1, in T2> 指向两个参数的方法

 Action<inT1, in T2, in T3, in T4, in T5, in T6, in T7, in T8> 指向八个参数的方法

 Func<T> 最多也可以传递16个参数,并且具有返回值类型

 Func<out TResult> 指向没有参数,有一个返回值类型的方法。

 Func<in T, out TResult> 指向一个参数的方法

 Func<in T1, inT2, in T3, in T4, out TResult> 指向一个具有四个参数的方法。

1、应用举例

在上面的实例中我们定义了一个代理来指向静态方法如下形式

delegate double DoubleOp(double x);

如果不想自己定义代理,可以使用如下形式来指向静态函数的两个方法

Func<double, double>[] operations =
{
 MathOperations.MultiplyByTwo,
 MathOperations.Square
};  //使用func<double, double> 来指向有返回值类型的一个参数。并不需要自己再自定代理,而使用系统自带的代理方法。

把这个代理作为参数传递给相应的方法,其格式如下

static void ProcessAndDisplayNumber(Func<double, double> action,double value) //第一个参数是代理实例,第二个参数是代理所用到的参数值
{
double result = action(value);
Console.WriteLine("Value is {0}, result of operation is {1}",
value, result);
}

原文地址:https://www.cnblogs.com/tianmochou/p/4933874.html