CLR via C# 委托 初识委托

Microsoft.NET Framework 通过委托来提供回调函数机制。不同于其他平台(比如非托管堆C++)的回调机制,委托的功能要多得多。例如,委托确保回调方法是类型安全的(这是CLR最重要的目标之一)。委托还允许顺序调用多个方法,并支持调用静态方法和实例方法。

using System;
using System.Windows.Forms;

namespace CLRviaCSharp
{
    //声明一个委托类型,它的实例引用一个方法,
    //该方法获取一个int参数,返回void
    internal delegate void Feedback(int value);

    class Program
    {
        private static void Counter(int from, int to, Feedback fb)
        {
            for (int i = from; i <= to; i++)
            {
                if (fb != null)
                {
                    fb(i);
                }
            }
        }

        private static void StaticFeedbackToConsole(int value)
        {
            Console.WriteLine("StaticFeedbackToConsole = " + value);
        }

        private void InstanceFeedbackToConsole(int value)
        {
            Console.WriteLine("InstanceFeedbackToConsole = " + value);
        }

        private static void FeedbackToMsgBox(int value)
        {
            MessageBox.Show("MessageBox = " + value);
        }

        private static void InstanceDelegateDemo()
        {
            Console.WriteLine("----InstanceDelegateDemo----");
            Program p = new Program();
            Counter(1, 3, new Feedback(p.InstanceFeedbackToConsole));
        }

        private static void StaticDelegateDemo()
        {
            Console.WriteLine("----StaticDelegateDemo----");
            Counter(1, 3, null);
            Counter(1, 3, new Feedback(Program.StaticFeedbackToConsole));
            Counter(1, 3, new Feedback(StaticFeedbackToConsole));
        }

        private static void ChainDelegateDemo1(Program p)
        {
            Console.WriteLine("----ChainDelegateDemo1----");
            Feedback fb1 = new Feedback(StaticFeedbackToConsole);
            Feedback fb2 = new Feedback(FeedbackToMsgBox);
            Feedback fb3 = new Feedback(p.InstanceFeedbackToConsole);

            Feedback fbChain = null;
            fbChain = (Feedback)Delegate.Combine(fbChain, fb1);
            fbChain = (Feedback)Delegate.Combine(fbChain, fb2);
            fbChain = (Feedback)Delegate.Combine(fbChain, fb3);

            Counter(1, 2, fbChain);

            Console.WriteLine();

            //fbChain = (Feedback)Delegate.Remove(fbChain, new Feedback(FeedbackToMsgBox));
            fbChain = (Feedback)Delegate.Remove(fbChain, fb2);

            Counter(1, 2, fbChain);
        }

        private static void ChainDelegateDemo2(Program p)
        {
            Console.WriteLine("----ChainDelegateDemo2----");
            Feedback fb1 = new Feedback(StaticFeedbackToConsole);
            Feedback fb2 = new Feedback(FeedbackToMsgBox);
            Feedback fb3 = new Feedback(p.InstanceFeedbackToConsole);

            Feedback fbChain = null;
            fbChain += fb1;
            fbChain += fb2;
            fbChain += fb3;

            Counter(1, 2, fbChain);

            Console.WriteLine();

            //fbChain -= new Feedback(FeedbackToMsgBox);
            fbChain -= fb2;

            Counter(1, 2, fbChain);
        }

        static void Main(string[] args)
        {

            StaticDelegateDemo();
            InstanceDelegateDemo();
            ChainDelegateDemo1(new Program());
            ChainDelegateDemo2(new Program());



















            Console.ReadLine();
        }
    }
}

在顶部,注意看internal委托Feedback的声明。委托要指定一个回调方法签名。在本例中,Feedback委托指定的方法要获取一个int参数,返回void。

//--用委托回调静态方法

示例代码中的StaticDelegateDemo方法。

第一次调用Counter方法时,参数为null,故不会调用。

第二次,参数传递新构造的Feedback稳妥对象。委托对象是方法的包装器(wrapper),使方法能通过包装器来间接回调。Program.StaticFeedbackToConsole被传给Feedback委托类型的构造器,这就是要包装的方法。new 操作符返回的引用作为Counter的第三个参数来传递。

注意:StaticFeedbackToConsole被定义成Program类型的内部的私有方法,但Counter方法能调用Program的私有方法。这明显没有问题,因为Counter和StaticFeedbackToConsole在同一个类型中定义。但即使Counter方法在另一个类型中定义,也不会出问题!简单地说,在一个类型中通过委托来调用另一个类型的私有成员,只要委托对象是由具有足够安全性/可访问性的代码创建的,便没有问题。

第三次与第二次几乎一致。

这个例子中的所有操作都是类型安全的。例如,在构造Feedback委托对象时,编译器确保Program的StaticFeedbackToConsole和FeedbackToMsgBox方法的签名兼容于Feedback委托定义的签名。具体地说,两个方法都要获取一个参数(一个int32),而且两者都要有相同的返回类型(void)。

将StaticFeedbackToConsole的定义改为下面这样:

将方法绑定到委托时,C#和CLR都允许引用类型的协变性(covariance)和逆变性(contravariance)。协变性是指方法能返回从委托的返回类型派生的一个类型。逆变性是指方法获取的参数可以是委托的参数类型的基类。

例如下面这个委托:

完全可以构造该委托类型的一个实例并绑定具有以下原型的方法:

注意,只有引用类型才支持协变性与逆变性,值类型或void不支持。

是因为它们的存储结构是变化的,而引用类型的存储结构始终是一个指针。

//--用委托回调实例方法

委托除了能调用静态方法,还能为具体的对象调用实例方法。

看示例代码中的InstanceDelegateDemo方法。

InstanceDelegateDemo方法构造了名为p的Program对象。这个Program对象仅为演示用。在Counter方法调用中构造新的Feedback委托对象,向Feedback委托类型的构造函数传递的是p.InstanceFeedbackToConsole这导致委托包装对InstanceFeedbackToConsole方法的引用,这是一个实例方法。当Counter调用由其fb实参标识的回调方法时,会调用InstanceFeedbackToConsole实例方法,新构造的对象p的地址作为隐式的this参数传给这个实例方法。

如果是实例方法,委托要知道方法操作的是具体哪个对象实例。包装实例方法很有用,因为对象内部的代码可以访问对象的实例成员。这意味着对象可以维护一些状态,并在回调方法执行期间利用这些状态信息。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值