1. 前言
2. 定义和特性
3. 关于事件的一个小例子
Bash
namespace Event1{internal class Program{static void Main(string[] args){var instance = new Demo(); instance.DemoEvent += (sender, args) =>{ Console.WriteLine("执行事件1!"); }; instance.DemoEvent += (sender, args) =>{ Console.WriteLine("执行事件2!");}; Console.WriteLine("*开始发起事件!"); instance.Raise(); Console.WriteLine("*事件执行完毕,继续下一项工作!");}}public class Demo{public event EventHandler DemoEvent;public void Raise(){try{this.DemoEvent?.Invoke(this, EventArgs.Empty); Console.WriteLine("所有的事件处理已经被执行!");}catch(Exception ex){ Console.WriteLine("事件处理中发生异常!", ex.Message);}}}}
Bash
*开始发起事件!执行事件1!执行事件2!所有的事件处理已经被执行!*事件执行完毕,继续下一项工作!
instance.DemoEvent += (sender, args) =>
{
Console.WriteLine("执行事件1!");
throw new Exception("执行事件1,错误");
};
## 结果如下 ##
*开始发起事件!
执行事件1!
事件处理中发生异常!
*事件执行完毕,继续下一项工作!
instance.DemoEvent += Instance_DemoEvent;
instance.DemoEvent-= Instance_DemoEvent;
4. 事件的异步处理
instance.DemoEvent += async (sender, args) =>
{
Console.WriteLine("执行事件1开始??");
await Task.Delay(10);
Console.WriteLine("执行事件1结束??");
};
// 为了等待这个慢家伙,需要在事件执行完毕的后面增加一行代码,让主程序等会退出。
Console.ReadLine();
*开始发起事件!
执行事件1开始??
执行事件1!
执行事件2!
所有的事件处理已经被执行!
*事件执行完毕,继续下一项工作!
执行事件1结束??
5. 等待所有异步处理订阅者
所有的事件处理已经被执行!
*事件执行完毕,继续下一项工作!
执行事件1结束??
public class NaiveSynchronizationContext : SynchronizationContext
{
private readonly Action completed;
public NaiveSynchronizationContext( Action completed)
{
this.completed = completed;
}
public override SynchronizationContext CreateCopy()
{
return new NaiveSynchronizationContext(this.completed);
}
public override void OperationStarted()
{
Console.WriteLine("同步上下文: 开始");
}
public override void OperationCompleted()
{
Console.WriteLine("同步上下文: 完成");
this.completed();
}
}
public static class NaiveExtension
{
public static Task NaiveRaiseAsync( this EventHandler @this, object sender, EventArgs eventArgs)
{
// 如果没有事件处理,那么立即结束
if (@this == )
{
return Task.CompletedTask;
}
var tcs = new TaskCompletionSource<bool>();
var sc = new NaiveSynchronizationContext(() => tcs.SetResult(true));
SynchronizationContext.SetSynchronizationContext(sc);
@this.Invoke(sender, eventArgs);
return tcs.Task;
}
}
public void Raise()
{
try
{
this.DemoEvent?.NaiveRaiseAsync(this,EventArgs.Empty).GetAwaiter().GetResult();
//this.DemoEvent?.Invoke(this, EventArgs.Empty);
Console.WriteLine("所有的事件处理已经被执行!");
}
catch(Exception ex)
{
Console.WriteLine("事件处理中发生异常!", ex.Message);
}
}
*开始发起事件!
同步上下文: 开始
执行事件1开始??
执行事件1!
执行事件2!
执行事件1结束??
同步上下文: 完成
所有的事件处理已经被执行!
*事件执行完毕,继续下一项工作!
System.InvalidOperationException:
“An attempt was made to transition a task to a final state when it had already completed.”
public static Task NaiveRaiseAsync( this EventHandler @this, object sender, EventArgs eventArgs)
{
// 如果没有事件处理,那么立即结束
if (@this == )
{
return Task.CompletedTask;
}
var delegates = @this.GetInvocationList();
var count = delegates.Length;
var tcs = new TaskCompletionSource<bool>();
foreach (var @delegate in @this.GetInvocationList())
{
// 检查AsyncStateMachineAttribute属性,判断是否异步处理函数
var async = @delegate.Method.GetCustomAttributes(typeof(AsyncStateMachineAttribute),false).Any();
// 定义 'completed' action
var completed = new Action(() =>
{
if (Interlocked.Decrement(ref count) == 0)
{
tcs.SetResult(true);
}
});
if (async)
{
SynchronizationContext.SetSynchronizationContext(new NaiveSynchronizationContext(completed));
}
@delegate.DynamicInvoke(sender, eventArgs);
if (!async)
{
// 如果不是异步,手工调用完成
completed();
}
}
return tcs.Task;
}
*开始发起事件!
同步上下文: 开始
执行事件1开始??
同步上下文: 开始
执行事件2开始??
执行事件1!
执行事件2!
执行事件2结束??
执行事件1结束??
同步上下文: 完成
同步上下文: 完成
所有的事件处理已经被执行!
*事件执行完毕,继续下一项工作!
6. 捕获异常处理中的异常
instance.DemoEvent += async (sender, args) =>
{
Console.WriteLine("执行事件1开始??");
throw new InvalidOperationException("Sabotage!");
await Task.Delay(10);
Console.WriteLine("执行事件1结束??");
};
*开始发起事件!
同步上下文: 开始
执行事件1开始??
同步上下文: 完成
同步上下文: 开始
Unhandled exception. 执行事件2开始??
执行事件1!
执行事件2!
执行事件2结束??
同步上下文: 完成
所有的事件处理已经被执行!
*事件执行完毕,继续下一项工作!
System.InvalidOperationException: Sabotage!
public class NaiveSynchronizationContext : SynchronizationContext
{
private readonly Action completed;
private readonly Action<Exception> failed;
public NaiveSynchronizationContext(Action completed, Action<Exception> failed)
{
this.completed = completed;
this.failed = failed;
}
public override void Post(SendOrPostCallback d, object state)
{
if (state is ExceptionDispatchInfo edi)
{
Console.WriteLine("正捕获异常");
this.failed(edi.SourceException);
}
else
{
Console.WriteLine("Posting");
base.Post(d, state);
}
}
public override void Send( SendOrPostCallback d, object state)
{
if (state is ExceptionDispatchInfo edi)
{
Console.WriteLine("正捕获异常");
this.failed(edi.SourceException);
}
else
{
Console.WriteLine("Sending");
base.Send(d, state);
}
}
public override SynchronizationContext CreateCopy()
{
return new NaiveSynchronizationContext(this.completed,this.failed);
}
public override void OperationStarted()
{
Console.WriteLine("同步上下文: 开始");
}
public override void OperationCompleted()
{
Console.WriteLine("同步上下文: 完成");
this.completed();
}
}
public static class NaiveExtension
{
public static Task NaiveRaiseAsync( this EventHandler @this, object sender, EventArgs eventArgs)
{
// 如果没有事件处理,那么立即结束
if (@this == )
{
return Task.CompletedTask;
}
var delegates = @this.GetInvocationList();
var count = delegates.Length;
var tcs = new TaskCompletionSource<bool>();
var exception = (Exception);
foreach (var @delegate in @this.GetInvocationList())
{
// 检查AsyncStateMachineAttribute属性,判断是否异步处理函数
var async = @delegate.Method.GetCustomAttributes(typeof(AsyncStateMachineAttribute),false).Any();
// 定义 'completed' action
var completed = new Action(() =>
{
if (Interlocked.Decrement(ref count) == 0)
{
if (exception is )
{
tcs.SetResult(true);
}
else
{
tcs.SetException(exception);
}
}
});
var failed = new Action<Exception>(e =>
{
Interlocked.CompareExchange( ref exception, e, );
});
if (async)
{
SynchronizationContext.SetSynchronizationContext(new NaiveSynchronizationContext(completed, failed));
}
ry
{
@delegate.DynamicInvoke(sender, eventArgs);
}
catch (TargetInvocationException e)
when (e.InnerException != )
{
failed(e.InnerException);
}
catch (Exception e)
{
failed(e);
}
if (!async)
{
// 如果不是异步,手工调用完成
completed();
}
}
return tcs.Task;
}
}
*开始发起事件!
同步上下文: 开始
执行事件1开始??
正捕获异常
同步上下文: 完成
同步上下文: 开始
执行事件2开始??
执行事件1!
执行事件2!
Posting
执行事件2结束??
同步上下文: 完成
事件处理中发生异常!
*事件执行完毕,继续下一项工作!