微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

c# – 始终取消Async CancellationTokenSource的最佳实践

所以我在我的UI上有一个组合框,它在SelectionChanged上异步地转到Web服务,以撤回一些要在UI上显示的信息(使用新的C#5 async / await关键字).我要做的是在发送新的异步请求之前取消当前的异步请求;例如,如果用户使用键盘快速循环遍历所有组合框项,则在第一个异步请求返回之前,SelectionChanged事件可能会多次触发(生成多个异步请求).

因此,从组合框的SelectionChanged事件中调用的异步函数如下所示:

public async Task<Connections> ConnectionsAsync()
{
    return await Task.Factory.StartNew(() => Connections,_cancellationTokenSource.Token);
}

Connections是一个关闭并触及Web服务的属性.因为一旦取消,CancellationTokenSource就无法重用,我想这样做:

public async Task<Connections> ConnectionsAsync()
{
    _cancellationTokenSource.Cancel();
    _cancellationTokenSource = new CancellationTokenSource();
    return await Task.Factory.StartNew(() => Connections,_cancellationTokenSource.Token);
}

但问题是,当没有运行异步命令时(例如第一次调用函数),有时我会调用Cancel();因此,如果我连接任何取消事件处理程序,它们将被调用,甚至在我发出异步请求之前.

无论如何都要检查异步请求是否已经在运行?除了我做的事情:

public async Task<Connections> ConnectionsAsync()
{
    if (_runningAsyncCommand)
        _cancellationTokenSource.Cancel();
    _cancellationTokenSource = new CancellationTokenSource();
    _runningAsyncCommand = true;
    return await Task.Factory.StartNew(() => Connections,_cancellationTokenSource.Token);
    _runningAsyncCommand = false;
}

我有一些异步函数都使用相同的CancellationTokenSource,所以我必须在所有这些函数中实现这个“管道”.这是最好的方法吗?或者,还有更好的方法

另外,如果我公开公开_cancellationTokenSource以便其他类可以使用它注册取消委托,那么将这些委托“转移”到新的CancellationTokenSource的最佳方法是什么,因为我每次都创建一个新的?

提前致谢!

解决方法

看起来像是在天堂进行Reactive Extensions的比赛.在创建任务之前观察ComboBox中的事件时,定义必须经过的节流时间(比如说300mS)

订阅TextBox更改事件的代码片段,但您会明白:

var input (from evt in Observable.FromEvent<EventArgs>(txt,"TextChanged")
select ((TextBox)evt.Sender).Text)
.Throttle(TimeSpan.FromSeconds(1))
.distinctUntilChanged();

using (input.Subscribe(inp => Console.WriteLine("You wrote: " + inp)))
{
  // Do stuff
}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。

相关推荐