我认为我在这里遗漏了一些明显的东西,但是在使用任务和检索值时如何更新GUI? (我正在尝试使用await / async而不是BackgroundWorker)
在我的控制下,用户点击了一个按钮,该按钮会做一些需要时间的事情。我想提醒父表单,以便它可以显示一些进展:
private void ButtonClicked()
{
var task = Task<bool>.Factory.StartNew(() =>
{
WorkStarted(this, new EventArgs());
Thread.Sleep(5000);
WorkComplete(this, null);
return true;
});
if (task.Result) MessageBox.Show("Success!");//this line causes app to block
}
在我的父表单中,我正在收听WorkStarted和WorkComplete来更新状态栏:
myControl.WorkStarting += (o, args) =>
{
Invoke((MethodInvoker) delegate
{
toolStripProgressBar1.Visible = true;
toolStripStatusLabel1.Text = "Busy";
});
};
如果我错了,请纠正我,但应用程序正在挂起,因为“Invoke”正在等待GUI线程变为可用,直到我的“ButtonClicked()”调用完成才会出现。所以我们陷入僵局。 解决这个问题的正确方法是什么?
答案 0 :(得分:2)
您正在阻止UI线程Task.Result阻止,直到任务完成。
试试这个。
private async void ButtonClicked()
{
var task = Task<bool>.Factory.StartNew(() =>
{
WorkStarted(this, new EventArgs());
Thread.Sleep(5000);
WorkComplete(this, null);
return true;
});
await task;//Wait Asynchronously
if (task.Result) MessageBox.Show("Success!");//this line causes app to block
}
答案 1 :(得分:1)
您可以使用Task.Run
在后台线程上执行代码。 Task-based Asynchronous Pattern指定进度更新的模式,如下所示:
private async void ButtonClicked()
{
var progress = new Progress<int>(update =>
{
// Apply "update" to the UI
});
var result = await Task.Run(() => DoWork(progress));
if (result) MessageBox.Show("Success!");
}
private static bool DoWork(IProgress<int> progress)
{
for (int i = 0; i != 5; ++i)
{
if (progress != null)
progress.Report(i);
Thread.Sleep(1000);
}
return true;
}
如果您的目标是.NET 4.0,那么您可以使用Microsoft.Bcl.Async
;在这种情况下,您必须使用TaskEx.Run
而不是Task.Run
。我在我的博客why you shouldn't use Task.Factory.StartNew
上解释。