博客
关于我
委托-利用GetInvocationList处理链式委托
阅读量:582 次
发布时间:2019-03-11

本文共 2591 字,大约阅读时间需要 8 分钟。

在C#中,当使用委托代理多个函数时,如果每次都使用+=来添加新的函数到同一个委托链中,可能会遇到一个问题:每次添加新的函数都会覆盖之前的委托,导致最终只能看到最后一个函数的返回值。为了正确地获取所有函数的返回值,可以使用GetInvocationList方法来遍历委托链中的所有函数。

以下是详细的解决方案:

  • 定义委托和类

    • 定义一个委托TestDelegate,其返回类型为string
    • 创建一个类DelegateClass,包含一个TestDelegate类型的属性testDelegate
  • 在Main方法中创建函数实例并添加到委托链

    • 创建四个不同的函数实例onetwothreefour,它们都实现了Say方法,返回不同的字符串。
    • 使用+=操作符将每个函数的Say方法添加到DelegateClasstestDelegate属性中。
  • 调用委托并遍历所有函数

    • DelegateClass类中实现InvokeDelegate方法。
    • InvokeDelegate方法中,检查testDelegate是否不为空。
    • 如果不为空,使用GetInvocationList方法获取委托链中的所有函数。
    • 遍历每个函数,逐个转换为TestDelegate类型并调用,获取返回值。
    • 打印每个返回值。
  • 完整代码示例

    using System;namespace GetInvocationListDemo{    public delegate string TestDelegate();    public class Program    {        static void Main(string[] args)        {            DelegateClass delegateClass = new DelegateClass();            TestMethodOne one = new TestMethodOne();            TestMethodTwo two = new TestMethodTwo();            TestMethodThree three = new TestMethodThree();            TestMethodFour four = new TestMethodFour();            delegateClass.testDelegate += one.Say;            delegateClass.testDelegate += two.Say;            delegateClass.testDelegate += three.Say;            delegateClass.testDelegate += four.Say;            delegateClass.InvokeDelegate();            Console.ReadKey();        }    }    public class DelegateClass    {        public TestDelegate testDelegate;        public void InvokeDelegate()        {            if (testDelegate != null)            {                foreach (Delegate dele in testDelegate.GetInvocationList())                {                    TestDelegate delegateClass = (TestDelegate)dele;                    string resultStr = delegateClass();                    Console.WriteLine(resultStr);                }            }        }    }    public class TestMethodOne    {        public string Say()        {            return "You called me from TestMethodOne~~~";        }    }    public class TestMethodTwo    {        public string Say()        {            return "You called me from TestMethodTwo~~~";        }    }    public class TestMethodThree    {        public string Say()        {            return "You called me from TestMethodThree~~~";        }    }    public class TestMethodFour    {        public string Say()        {            return "You called me from TestMethodFour~~~";        }    }}
  • 运行结果

    • 执行上述代码时,会依次输出四个方法的返回值:
      You called me from TestMethodOne~~~You called me from TestMethodTwo~~~You called me from TestMethodThree~~~You called me from TestMethodFour~~~
  • 通过使用GetInvocationList方法,可以确保所有添加到委托链中的函数都会被调用并获取其返回值,从而避免只保留最后一个函数的结果。

    转载地址:http://pdlvz.baihongyu.com/

    你可能感兴趣的文章
    orm总结
    查看>>
    os.path.join、dirname、splitext、split、makedirs、getcwd、listdir、sep等的用法
    查看>>
    os.system 在 Python 中不起作用
    查看>>
    OSCACHE介绍
    查看>>
    SQL--合计函数(Aggregate functions):avg,count,first,last,max,min,sum
    查看>>
    OSChina 周四乱弹 ——程序员为啥要买苹果手机啊?
    查看>>
    OSError: no library called “cairo-2“ was foundno library called “cairo“ was foundno library called
    查看>>
    OSG学习:几何体的操作(二)——交互事件、Delaunay三角网绘制
    查看>>
    OSG学习:几何对象的绘制(三)——几何元素的存储和几何体的绘制方法
    查看>>
    OSG学习:几何对象的绘制(二)——简易房屋
    查看>>
    OSG学习:场景图形管理(一)——视图与相机
    查看>>
    OSG学习:场景图形管理(三)——多视图相机渲染
    查看>>
    OSG学习:场景图形管理(二)——单窗口多相机渲染
    查看>>
    OSG学习:场景图形管理(四)——多视图多窗口渲染
    查看>>
    OSG学习:新建C++/CLI工程并读取模型(C++/CLI)——根据OSG官方示例代码初步理解其方法
    查看>>
    Sql 随机更新一条数据返回更新数据的ID编号
    查看>>
    OSG学习:空间变换节点和开关节点示例
    查看>>
    OSG学习:纹理映射(一)——多重纹理映射
    查看>>
    OSG学习:纹理映射(七)——聚光灯
    查看>>
    OSG学习:纹理映射(三)——立方图纹理映射
    查看>>