非同期WPFコマンド
-
02-07-2019 - |
質問
注 :この質問のコードは deSleeper 。
コマンドで欲しかったものの1つは、非同期操作のベイクされたデザインでした。コマンドの実行中にボタンを押して無効にし、完了したら戻るようにしたかったのです。実際の作業をThreadPoolワークアイテムで実行したかったのです。最後に、非同期処理中に発生したエラーを処理する方法が必要でした。
私のソリューションはAsyncCommandでした:
public abstract class AsyncCommand : ICommand
{
public event EventHandler CanExecuteChanged;
public event EventHandler ExecutionStarting;
public event EventHandler<AsyncCommandCompleteEventArgs> ExecutionComplete;
public abstract string Text { get; }
private bool _isExecuting;
public bool IsExecuting
{
get { return _isExecuting; }
private set
{
_isExecuting = value;
if (CanExecuteChanged != null)
CanExecuteChanged(this, EventArgs.Empty);
}
}
protected abstract void OnExecute(object parameter);
public void Execute(object parameter)
{
try
{
IsExecuting = true;
if (ExecutionStarting != null)
ExecutionStarting(this, EventArgs.Empty);
var dispatcher = Dispatcher.CurrentDispatcher;
ThreadPool.QueueUserWorkItem(
obj =>
{
try
{
OnExecute(parameter);
if (ExecutionComplete != null)
dispatcher.Invoke(DispatcherPriority.Normal,
ExecutionComplete, this,
new AsyncCommandCompleteEventArgs(null));
}
catch (Exception ex)
{
if (ExecutionComplete != null)
dispatcher.Invoke(DispatcherPriority.Normal,
ExecutionComplete, this,
new AsyncCommandCompleteEventArgs(ex));
}
finally
{
dispatcher.Invoke(DispatcherPriority.Normal,
new Action(() => IsExecuting = false));
}
});
}
catch (Exception ex)
{
IsExecuting = false;
if (ExecutionComplete != null)
ExecutionComplete(this, new AsyncCommandCompleteEventArgs(ex));
}
}
public virtual bool CanExecute(object parameter)
{
return !IsExecuting;
}
}
だから質問は次のとおりです。これはすべて必要ですか?データバインディングの非同期サポートが組み込まれていることに気付いたので、コマンドを実行しないのはなぜですか。おそらく、次の質問であるパラメーターの質問に関連しているのでしょう。
解決
元のサンプルを絞り込んで、同様の状況に陥った他の人にアドバイスをすることができました。
まず、BackgroundWorkerがニーズを満たすかどうかを検討します。私はまだ自動無効化機能を取得するためにAsyncCommandを頻繁に使用していますが、BackgroundWorkerで多くのことができる場合は
しかし、BackgroundWorkerをラップすることにより、AsyncCommandはコマンドのような機能と非同期動作を提供します(このトピックに関するブログエントリ)
public abstract class AsyncCommand : ICommand
{
public event EventHandler CanExecuteChanged;
public event EventHandler RunWorkerStarting;
public event RunWorkerCompletedEventHandler RunWorkerCompleted;
public abstract string Text { get; }
private bool _isExecuting;
public bool IsExecuting
{
get { return _isExecuting; }
private set
{
_isExecuting = value;
if (CanExecuteChanged != null)
CanExecuteChanged(this, EventArgs.Empty);
}
}
protected abstract void OnExecute(object parameter);
public void Execute(object parameter)
{
try
{
onRunWorkerStarting();
var worker = new BackgroundWorker();
worker.DoWork += ((sender, e) => OnExecute(e.Argument));
worker.RunWorkerCompleted += ((sender, e) => onRunWorkerCompleted(e));
worker.RunWorkerAsync(parameter);
}
catch (Exception ex)
{
onRunWorkerCompleted(new RunWorkerCompletedEventArgs(null, ex, true));
}
}
private void onRunWorkerStarting()
{
IsExecuting = true;
if (RunWorkerStarting != null)
RunWorkerStarting(this, EventArgs.Empty);
}
private void onRunWorkerCompleted(RunWorkerCompletedEventArgs e)
{
IsExecuting = false;
if (RunWorkerCompleted != null)
RunWorkerCompleted(this, e);
}
public virtual bool CanExecute(object parameter)
{
return !IsExecuting;
}
}
他のヒント
他の質問で答えたように、おそらくこれに同期してバインドしてから、コマンドを非同期に起動したいでしょう。そうすることで、現在抱えている問題を回避できます。
所属していません StackOverflow