如何注销和注册不可用的事件处理程序

本文关键字:程序 事件处理 注册 何注销 注销 | 更新日期: 2023-09-27 18:24:10

在我的代码中,我需要注销和注册一个事件处理程序,这非常有效:

_favoritsImageView.Click -= _favoritsImageView_Click(this, new CustomeClickEventArgs(item));
_favoritsImageView.Click += _favoritsImageView_Click(this, new CustomeClickEventArgs(item));
void _favoritsImageView_Click(object sender, CustomeClickEventArgs e)
{
       // handles the event
}

但对于不可用的事件处理程序,我必须使用以下语法:

_favoritsImageView.Click -= async (s, e) => 
{ await _favoritsImageView_ClickAsync(s, new CustomeClickEventArgs(item)); };
_favoritsImageView.Click += async (s, e) => 
{ await _favoritsImageView_ClickAsync(s, new CustomeClickEventArgs(item)); };
async Task _favoritsImageView_ClickAsync(object sender, CustomeClickEventArgs e)
{
       // does async task
}

这不起作用。因为匿名方法没有相同的引用。因此,第一行并没有取消注册已经注册的处理程序。最后,第二行为单击添加了一个额外的事件处理程序。

我需要使用哪种语法来添加和删除异步事件处理程序

谢谢你的建议。

如何注销和注册不可用的事件处理程序

我需要使用哪种语法来添加和删除异步事件处理程序?

与您需要在常规事件处理程序中使用的语法相同。您需要将委托保存在某个位置,以便以后可以取消注册:

private EventHandler eventHandler = 
                            new EventHandler(async (s, e) => await FooAsync(s, e));
public async void SomeOtherEventHandler()
{
    var m = new M();
    m.X += eventHandler;
    m.OnFoo();
    m.X -= eventHandler;
    m.OnFoo();
}
public async Task FooAsync(object sender, EventArgs e)
{
    await Task.Delay(1000);
    Debug.WriteLine("Yay event handler");
}
public class M
{
    public event EventHandler X;
    public void OnX()
    {
        // If you're using C#-6, then X?.Invoke(null, EventArgs.Empty);
        var localX = X;
        if (localX != null)
            localX(null, EventArgs.Empty);
    }
}

编辑

@svick提出了另一种解决方案,简单地将方法async void直接注册,这肯定更短:

public async void SomeOtherEventHandler()
{
    var m = new M();
    m.X += FooAsync;
    m.OnFoo();
    m.X -= FooAsync;
    m.OnFoo();
}
public async void FooAsync(object sender, EventArgs e)
{
    await Task.Delay(1000);
    Debug.WriteLine("Yay event handler");
}

同步代码看起来像这样:

private void FavoritsImageView_Click(object sender, CustomeClickEventArgs args)
{
    // your synchronous code here
}
…
_favoritsImageView.Click += FavoritsImageView_Click;
_favoritsImageView.Click -= FavoritsImageView_Click;

async版本看起来几乎相同,您只需要将async添加到方法中:

private async void FavoritsImageView_Click(object sender, CustomeClickEventArgs args)
{
    // your asynchronous code here
}
…
_favoritsImageView.Click += FavoritsImageView_Click;
_favoritsImageView.Click -= FavoritsImageView_Click;

请注意,事件处理程序几乎是唯一应该使用async void的地方。在大多数其他情况下,同步void方法应该转换为async Task方法。