最近发现C#的事件和wpf的dispatcherobject在一起使用会有一些不容易发觉的问题。
我们都知道C#的事件原理,实际上是存储了一系列方法的委托。当事件引发的时候,依次调用(Invoke
)委托列表的委托进行执行。
所以从中可以发现显而易见的一些问题比如:
- 监听事件执行顺序无法保证
- 耗时委托执行拖慢其他业务注册的方法
- 资源泄露问题
这些很多人都会聊,我们就不讲了~
今天重点讲wpf会遇到的跨线程访问的问题。
我们都知道wpf的DispatcherObject,必须在创建它的Dispatcher上执行,而由于C#的事件机制,这个调用线程(及关联的Dispatcher)的控制权交给了事件引发者。
所以不注意的小伙伴就常常会出现这样的情况:
- 事件引发者时而从主线程
Invoke
,时而从后台线程Invoke
。 - 事件监听者概率性出现UI元素跨线程调用问题。
怎么办呢?
- 方案1:部分小伙伴会选择直接在事件注册的函数里添加
Application.Current.Dispatcher.Invoke
AppFoo.Login += ()=>
{
Application.Current.Dispatcher.Invoke(()=>
{
//业务
});
}
很不错,这个方法可以很好的解决问题。但是只能发现一处解决一处
- 方案2:于是又小伙伴别出心裁,在引发处处理
Application.Current.Dispatcher.Invoke(()=>
{
Login?.Invoke();
});
这样一来所有的注册函数都会在主线程执行。跨线程调用是解决了,但是UI卡顿就加重了。。
- 最佳实践:我们在事件注册函数中不相信那些事件引发者,(路由事件等确定是UI线程引发的事件除外),针对UI元素调用使用
Dispatcher.Invoke
,针对耗时操作使用异步方法。
一个附加的测试小案例,下面的代码小伙伴为了解决登陆事件早于注册时机的问题,在注册事件的时候判断是否已经登陆,如果是,则直接Invoke
函数
你们能够看出问题所在么?
private event EventHandler _userLogined;
public event EventHandler UserLogined
{
add
{
if (IsAuthed)
{
value.Invoke(this, EventArgs.Empty);
}
_userLogined += value;
}
remove
{
_userLogined -= value;
}
}
。
。。
。。。
答案揭晓:
- 注册的时候调用,执行的线程控制权就交给了注册方法。如果事件引发是来自主线程,注册的方法也是UI相关,那么如果在登陆后在后台线程注册,就会出现跨线程访问问题哦
本文会经常更新,请阅读原文: https://xinyuehtx.github.io/post/%E4%B8%8D%E8%A6%81%E7%9B%B8%E4%BF%A1%E9%82%A3%E4%BA%9B%E4%BA%8B%E4%BB%B6%E5%BC%95%E5%8F%91%E8%80%85.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名黄腾霄(包含链接: https://xinyuehtx.github.io ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 。