当我用linq键入时运行任务(如果仍在运行则取消上一个任务)

时间:2014-08-12 07:23:31

标签: c# linq entity-framework task cancellation-token

我想创建一个在我键入时使用linq搜索的任务,如果用户键入另一个字符,它应该取消任务并重新创建搜索,我有以下代码:

private Task SearchChannels;
private CancellationTokenSource cancelSearch;

public void PopulateChannels(string newValue)
{
    IsSearchingChannels = true; //This just shows a progressbar
    if (SearchChannels != null && cancelSearch!= null)
        if (SearchChannels.Status == TaskStatus.Running || 
            SearchChannels.Status == TaskStatus.WaitingToRun || 
            SearchChannels.Status == TaskStatus.WaitingForActivation || 
            SearchChannels.Status == TaskStatus.WaitingForChildrenToComplete) 
        {
            cancelSearch.Cancel();
            SearchChannels.Wait();
        }
    cancelSearch = new CancellationTokenSource();
    SearchChannels = new Task(() => Channels = new PagedObservableCollection<Channel>(ContractManager.Channels.Where(x => x.Name.ToLower().StartsWith(newValue)).AsParallel().WithCancellation(cancelSearch.Token).ToList()), cancelSearch.Token); //PagedObservableCollection is just a simple class with a list that keeps all items and an ObservableCollection for current items shown

    SearchChannels.Start();
    SearchChannels.ContinueWith((continuation) => IsSearchingChannels = false); // this just hides the progressbar when done
}

我得到了这个例外:

'System.OperationCanceledException'类型的例外 发生在System.Core.dll但未在用户代码中处理

其他信息:操作已取消。

我是一个有任务和cancelTokens的初学者,任何人都可以从这里指导我正确的道路吗?我基本上希望Task检查它是否已经运行,取消它,然后使用新值再次运行它(我想使这个“SearchBox”功能类似于Visual Studio在解决方案资源管理器中的搜索,在您键入时搜索)

1 个答案:

答案 0 :(得分:1)

首先,您需要创建一个IObservable<string>来抽象更改控件上的值。执行此操作的“最简单”方法是使用Subject<string>,但很可能是错误的方法。

以下是您应放入ViewModel的代码。

IDisposable _searchSubscriber =
    _searchString
         .Buffer(TimeSpan.FromMillisecond(300))
         .Select(searchString => 
                Observable.StartAsync(cancelToken => 
                      Search(searchString, cancelToken)
                ).Switch()
         .ObserveOn(new DispatcherScheduler())
         .Subscribe(results => Channels = results);

public Task<List<Channel>> Search(string searchTerm, CancellationToken cancel)
{
    var query = dbContext.Channels.Where(x => x.Name.StartsWith(searchTerm));
    return query.ToListAsync(cancel);
}

private BehaviorSubject<string> _searchString = new BehaviorSubject<string>("");
public string SearchString
{
    get { return _searchString.Value; }
    set { _searchString.OnNext(value); OnPropertyChanged("SearchString"); }
}

Rx.net是一个非常强大的库,当然这意味着它确实有一点学习曲线(尽管事实是这很复杂,因为你的问题很复杂)。

让我说出来......

.Buffer(TimeSpan.FromMilliseconds(300))会对您的查询进行去抖动,因此它只会每300毫秒运行一次查询。

Observable.StartAsync(cancelToken => Search(searchString, cancelToken))为搜索任务创建一个Observable,当它被处理时将被取消。

Select(x => ...).Switch()仅获取最新的查询结果,并处理最后一个查询。

ObserveOn(...)在使用的调度程序上运行以下命令,如果您使用DispatchScheduler,请务必使用WPF,如果使用Winforms,请确保使用WinformsScheduler

Subscribe(results => ...)对结果采取措施。

相关问题