使用MVVM在relaycommand中运行异步方法

时间:2015-07-23 09:09:47

标签: c# wpf mvvm async-await

我正在使用WPF和MVVM开发异步应用程序,但我似乎无法在我的relaycommand中运行异步方法。

我的WPF 视图上有一个按钮,连接到我的 viewmodel 中的relaycommand,它试图在我的模型中调用异步方法返回结果列表:

/// <summary>
/// Search results 
/// </summary>
private ObservableCollection<string> _searchResults = new ObservableCollection<string>(); 
public IList<string> SearchResults
{
    get { return _searchResults; }
}

/// <summary>
/// Search button command
/// </summary>
private ICommand _searchCommand;
public ICommand SearchCommand
{
    get
    {
        _searchCommand = new RelayCommand(
            async() =>
            {
                SearchResults.Clear();
                var results = await DockFindModel.SearchAsync(_selectedSearchableLayer, _searchString);
                foreach (var item in results)
                {
                    SearchResults.Add(item);
                }                    
                //notify results have changed
                NotifyPropertyChanged(() => SearchResults);
            },
            () => bAppRunning); //command will only execute if app is running
        return _searchCommand;
    }
}

但是当relay命令尝试执行时,我得到以下异常:

  

未处理的类型&#39; System.AggregateException&#39;发生在mscorlib.dll

     
    

附加信息:未发现任务的例外情况     通过等待任务或访问其Exception属性。作为一个     结果,终结器线程重新抛出了未观察到的异常。

  

我已经尝试了this thread中的许多内容来尝试解决问题但没有运气。有谁知道如何解决这个问题?

3 个答案:

答案 0 :(得分:3)

不确定RelayCommand的来源(MVVM框架或自定义实现),但请考虑使用异步版本。

public class AsyncRelayCommand : ICommand
{
    private readonly Func<object, Task> execute;
    private readonly Func<object, bool> canExecute;

    private long isExecuting;

    public AsyncRelayCommand(Func<object, Task> execute, Func<object, bool> canExecute = null)
    {
        this.execute = execute;
        this.canExecute = canExecute ?? (o => true);
    }

    public event EventHandler CanExecuteChanged
    {
        add { CommandManager.RequerySuggested += value; }
        remove { CommandManager.RequerySuggested -= value; }
    }

    public void RaiseCanExecuteChanged()
    {
        CommandManager.InvalidateRequerySuggested();
    }

    public bool CanExecute(object parameter)
    {
        if (Interlocked.Read(ref isExecuting) != 0)
            return false;

        return canExecute(parameter);
    }

    public async void Execute(object parameter)
    {
        Interlocked.Exchange(ref isExecuting, 1);
        RaiseCanExecuteChanged();

        try
        {
            await execute(parameter);
        }
        finally
        {
            Interlocked.Exchange(ref isExecuting, 0);
            RaiseCanExecuteChanged();
        }
    }
}

答案 1 :(得分:0)

好的,所以我实际上设法通过更改我运行异步方法的方式来解决问题。

我改变了这个:

var results = await DockFindModel.SearchAsync();

对此:

var results = await QueuedTask.Run(() => DockFindModel.SearchAsync());

虽然当relaycommand已经接受异步lambda时,我有点想知道为什么我需要等待Task.Run()。我确定它会随着时间变得清晰。

感谢那些评论的人。

答案 2 :(得分:0)

RelayCommand类可能正在接受类型为Action的参数。由于您传递的是异步lambda,因此我们这里有一个async void场景。 您的代码将被触发并忘记。考虑使用以Func<Task>作为参数而不是Action作为参数的ICommand实现。