C#中的事件是否结构化?

时间:2015-03-04 17:49:30

标签: c# events

所以我有一个EventHandlers字典,但我发现当我在将keyvaluepair添加到字典之前附加到一个事件时,一切正常。但是,如果我添加keyvaluepair然后更新eventhandler的值,则字典不会更新。

public static event EventHandler TestEvent;
private static Dictionary<int, EventHandler> EventMapping = new Dictionary<int, EventHandler>();

 //TestEvent += GTKWavePipeClient_TestEvent;

  EventMapping.Add(0, TestEvent);
  TestEvent += GTKWavePipeClient_TestEvent;
  //test event is non null now. keyvaluepair in EventMapping has a value of null

2 个答案:

答案 0 :(得分:7)

EventHandler之类的委托类型是不可变类型。使用赋值(=)或复合赋值(+=)时,会创建一个新实例。

字典保留旧实例。

委托类型是引用类型,但重要的是它们的不变性。

当您拥有event时,使用+=语法甚至不是一项任务。它是add访问者或事件的调用。它将以线程安全的方式重新分配后备字段(新实例)。


请记住,您可以自己创作活动访问者。例如:

public static event EventHandler TestEvent
{
  add
  {
    lock (lockObj)
    {
      EventHandler oldDel;
      if (EventMapping.TryGetValue(0, out oldDel))
        EventMapping[0] = oldDel + value;
      else
        EventMapping.Add(0, value);
    }
  }

  remove
  {
    lock (lockObj)
    {
      EventHandler oldDel;
      if (EventMapping.TryGetValue(0, out oldDel))
        EventMapping[0] = oldDel - value;
    }
  }
}
private static readonly object lockObj = new object();
private static Dictionary<int, EventHandler> EventMapping = new Dictionary<int, EventHandler>();

使用该代码,当你去:

TestEvent += GTKWavePipeClient_TestEvent;

您的add访问者使用&#34;隐式&#34;参数EventHandler value设置为GTKWavePipeClient_TestEvent

答案 1 :(得分:4)

代表是不可变的。在调用+ =附加事件时,您正在为TestEvent分配新对象。因此,在非工作场景中,词典中的对象不同于具有附加事件的对象。