我想创建一个在我键入时使用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在解决方案资源管理器中的搜索,在您键入时搜索)
答案 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 => ...)
对结果采取措施。