理解C#中的代理和事件(一)

类别:.NET开发 点击:0 评论:0 推荐:

解C#中的代理和事件(一)

  突然写这篇文章,真的有点,是在做作的感觉,我想这并不是什么,难以

理解的东西,事实上,很多人都写过,而且,我保证至少不比我写的差.可是

还是觉得有必要提出来.因为要去正确的理解代理和事件是很有趣的,也是

很必要的.那么好吧,下面我就来讲讲,它们之间的关系.当然还会有些例子.

  首先我想说说有关事件.事件顾名思义当然是windows程序收到的消息.

那么我举几个有关事件的例子,如鼠标移动.按下.之类的都是.那么代理呢?

很多人都说它看上去,就是想是一个受托管的函数指针.我觉得这种说法很

正确,事实如此.在MSDN上,我们经常可以看到,有关代理的"多路广播"这个词

我觉得不错,可是不好理解.但我会在下面详细讲解的.


代理:(有的书上也翻译成指代或委托.英文是这样一来的"Delegate")

我想很多刚接触C#的人,都会对代理产生兴趣的.事实上也是如此,不了解它,你就没办法

来做windows程序.和传统意义上的函数指针有所不同的是,代理在C#中是一种类型,这样

它看上去,更安全也,更符合OO精神.代理实际上所做的工作就是通过引用把函数包裹起来

并使函数有一个有效的返回值.不知道我这样说是否好理解.那么我举个例子,你去建造房子.

很显然,我是在说你所做的事情.那么建造房子就是代理,它指代了,你要做的事情,可是它并

没有去做任何事情,事实上是,在建造房子这个工作里,你做了,那么结果是什么?当然是建立

一座房子.是的,建造房子就是代理,而如何建造房子则是函数应该完成的工作.而建造的是什么样

的房子,则是返回值.还记得,我曾经说过,代理是一种类型吗?呵呵..我想你应该记得,因为,

那是很新颖的,至少当时我那么认为.好吧,让我们来看看名称空间System.Delagate,看见了吗?那

就代理类.

举个例子:
public delegate void GetString()//我申明了一个代理

现在我要用到它了如下;
int i=100;
GetString gs=new GetString(i.ToString);//这里我吧int的ToString方法

填入了一个代理中.看上去想构造函数.这就是常在书上看到的"名称等效的,而

不是结构等效的",我想看到这儿你还是不明白.那么,我再来一个代理

如下:

float j=0.0001;

GetString gs=new GetString(j.ToString);//瞧见了,int的ToString方法

和float的ToString方法的结构是不一样的,可是名称和类型的返回值和

参数都一样.现在,我想,你应该理解了吧.


可是,我们经常会在MSDN中看到,这样的句子.单路代理和多路广播.看上去,有点不好理解.

事实上,我开始看这样的句子,是有点头痛的.那么,我想例子是最好的解说方式.

Single Delegate:(单路代理)

从字面上,我们可以这样来理解,这个代理只是单单代理了一个函数的工作.那么好吧,让

我们来看看它是如何工作的.下面我就来定义一个这样的代理:

public delegate bool Myfun(string str,int i)


现在我再来写一个方法如下:
 
bool CompareStrToInt(string s,int i)
{
   if(s.CompareTo(i.ToString())==0)
        return true;
    else
         return false;
}

这个方法完成的工作很简单对吧,只是比较字符而已.那么和代理有什么关系呢?还记得

我说的话吗?代理就是在把动词名词化.代码如下:

Myfun mf=new  (CompareStrToInt);

string s="10000";
int i=10000;
ConSole.WriteLine("Value="+mf(s,i));

输出结果:

Value=true

这就是单路代理.它只代理一个.好吧,也许你想看看复杂的例子,更有趣的在后面呢,

该是讨论多路广播的时候了.


多路广播:


一个代理同时代理几个方法.就是我们前面说到的那样,你去建造房子,现在要不仅仅是

建造住宅,还的去建造花园等等其它建筑物.可是它们都是在建造房子,传递的参数也相同

返回值的类型也相同都是房屋.那么我们为什么不找一个代理人来完成这样的任务呢?把

这些事物交由他一个人来完成不是可以节省我们很多的时间和金钱.是的我们可以那样做

System.MulticastDelegate 实际上在.net framework中你还可以找到这个类,多路代理

MSDN上翻译成多路广播.事实上它还重载了操作符+=.其实多路广播和单路代理在使用方法

上区别不大.你可以看下面的例子.


using System;

namespace Multi_castDelegate
{
  /// <summary>
  /// Summary description for Class1.
  /// </summary>
  class MyClassDelegate
  {
    /// <summary>
    /// The main entry point for the application.
    /// </summary>
   
    public delegate string IntDelegate(string s);
  }
}
using System;

namespace Multi_castDelegate
{
  /// <summary>
  /// Summary description for MyImplementingClass.
  /// </summary>
  public class MyClass
  {
    public MyClass()
    {
     
    }

    public static string WriteString(string s)
    {
      Console.WriteLine("Writing string");
      return "null";
    }

    public static string logString(string s)
    {
      Console.WriteLine("loging string"); 
      return "null";
    }

    public static string TransmitString(string s)
    {
      Console.WriteLine("Transmitting string"); 
      return "null";
      
    }
  }
}

The Main class:
using System;
using System.Threading;
namespace Multi_castDelegate
{
  /// <summary>
  /// Summary description for Test.
  /// </summary>
  public class Test
  {
    public static void Main()
    {
      MyClassDelegate.StringDelegate
        Writer,Logger,Transmitter;

      MyClassDelegate.StringDelegate
        myDelegate;

      Writer=new
                   MyClassDelegate.StringDelegate(MyClass.WriteString);
      /// calling Writer
      Writer("hello i am Writer just acting like Single cast");
      Logger=new MyClassDelegate.StringDelegate(MyClass.logString);
      ///calling Logger
      Logger("hello i am Logger just acting like Single-cast");
      Transmitter=new MyClassDelegate.StringDelegate(MyClass.TransmitString);
      ///calling Transmitter
            Transmitter("hello i am Transmitter just acting like Single-cast"); 
      ///here mydelegate used the Combine method of System.MulticastDelegate
      ///and the delegates combine  
      myDelegate=(MyClassDelegate.StringDelegate)System.Delegate.Combine(Writer,Logger); 
            myDelegate("used Combine");
      ///here Transmitter is also added using the overloaded form of Combine
      myDelegate+=Transmitter;
      myDelegate("Using Overloaded Form");
      ///now using the Remove method
      myDelegate=(MyClassDelegate.StringDelegate)System.Delegate.Remove(myDelegate,Writer);
            myDelegate("Without Writer");    
      ///overloaded Remove
      myDelegate-=Transmitter;
      myDelegate("Without Transmitter");
        System.Threading.Thread.Sleep(2300);  
           
    }
  }
}

(上面的例子是在一个国外网站上找到的,觉得不错,就直接套用了.)

上面的例子重点是看那两个已经重载的操作符."-="和"+=".通过上面的例子,你可以清楚的

看到多路广播是如何一次代理多个方法的.当然你也可以删除掉那些你不想要的用"-="操作

符就可以了.(那么我将在下一篇讨论事件)

本文地址:http://com.8s8s.com/it/it45821.htm