所以我有一个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
Run Code Online (Sandbox Code Playgroud)
委托类型EventHandler是不可变类型.使用assignment(=)或复合赋值(+=)时,会创建一个新实例.
字典保留旧实例.
委托类型是引用类型,但重要的是它们的不变性.
当你有一个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>();
Run Code Online (Sandbox Code Playgroud)
使用该代码,当你去:
TestEvent += GTKWavePipeClient_TestEvent;
Run Code Online (Sandbox Code Playgroud)
add调用您的访问器时将"隐式"参数EventHandler value设置为GTKWavePipeClient_TestEvent.