c#委托中另外一种用法

    在c#委托中,经常可能遇到函数重载的情况,可是又需要在一个函数中调用这些函数,一般我都是根据多个函数重载个数,也写上这么多个函数重载。比如

public double T1(int r)
{
   return r;
}
public double T2(double n1, double n2)
{
 return n1 + n2;
}

    现在呢,我需要写另外一个函数,这个函数第一个参数就是委托类型,我想让他既可以代表T1,也可以代表T2,但是一般是无法这样做的,因此我们只能写两个函数

        public void DT1(Func<double,int> fun,double r)
        {
            Console.WriteLine("这是T1方法得到的结果:{0}",fun(r));
        }
        public void DT1(Func<double,double,double>fun, double n1,double n2)
        {
            Console.WriteLine("这是T1方法得到的结果:{0}", fun(n1,n2));
        }

    当函数少的时候这样写还可以,一旦多了呢,或者根本不知道要传哪种类型的委托,这时候该怎么办,我们想到了多态,也就是传一个委托的父类,参数声明为object类型,比如:

        public void T(Delegate d,object[] pms)
        {
        }

     但是,这样该如何调用呢,我们首先想到那我能不能直接d(pms)呢, 答案肯定是不行的,如果你这样做,编译器就会提示:”“d”是“变量”,但此处被当做“方法”来使用”,为什么会出现这种情况,我们转到这个类的定义查看,发现Delegate是一个abstract 类,也就是只有在执行的过程中,CLR才会知道到底是d代表的是哪种委托,所以就无法这样做了~~。然后我们想到了反射,是不是可以利用反射呢。果然,这个类下面有一个属性 public MethodInfo Method { get; } 这个属性是一个MethodInfo类型的。于是我们这样改一下代码:

public void T(Delegate d,object[] pms)
  {
            //d(pms);
            double r = 0;
            System.Reflection.MethodInfo method = d.Method;
            r=(double)method.Invoke(this, pms);
            Console.WriteLine("Invoke得到的结果:{0}",r);
  }

    invoke中的第一个参数是这个方法是哪个类的,如果是静态方法,则忽略此参数,第二个参数是一个object数组,代表了这个委托代表方法的参数,按顺序传递。Invoke方法的返回值就是执行方法的返回值,这里强转一下,就能得到结果。

static void Main(string[] args)
        {
            Test ct = new Test();
            ct.T(new Func<int, double>(ct.T1), new object[] { 1 });
            ct.T(new Func<double, double, double>(ct.T2), new object[] { 1, 2 });
            Console.ReadKey();
        }

执行结果为:

Invoke得到的结果:1
Invoke得到的结果:3

    Delegate中还有一个方法,protected virtual object DynamicInvokeImpl(object[] args) 这是一个虚函数,和Invoke方法是一样的,只不过省略了第一个参数

        public void T(Delegate d,object[] pms)
        {
            //d(pms);
            double r = 0;
            //System.Reflection.MethodInfo method = d.Method;
            //r=(double)method.Invoke(this, pms);
             r=(double)d.DynamicInvoke(pms);
            Console.WriteLine("Invoke得到的结果:{0}",r);
        }

    这样也能达到同样的效果。

    第一次写博客。。有的东西也是似懂非懂,还请各位大神见谅。。。

原文地址:https://www.cnblogs.com/wzxwhd/p/5859023.html