模态对话框中的TPL任务进度

时间:2013-02-05 19:44:03

标签: c# winforms task-parallel-library

假设我有一个我想要运行的长时间运行的任务,并且可以通知用户界面其状态/进度。这里的问题是我想要的UI是一个模态对话框,我在开始任务之前无法显示它,也无法启动任务然后显示它。

尝试1:首先显示对话框,然后启动任务:

ProgressDialog dlg = new ProgressDialog();
Task longRunningTask = CreateTask();  
dlg.ShowDialog();
longRunningTask.Start();
Task.WaitAll(longRunningTask);

这显然不起作用,线程在ShowDialog()被阻止,任务永远不会启动。

尝试2:启动任务,然后显示对话框:

ProgressDialog dlg = new ProgressDialog();
Task longRunningTask = CreateTask();  
longRunningTask.Start();
dlg.ShowDialog();
Task.WaitAll(longRunningTask);

但在这种情况下,我冒险从任务中丢失第一个进度事件,或者更糟糕的是,当“InvokeRequired”显示为false时,我得到了可怕的跨线程异常,因为第一个事件在创建时到达进度对话框,但已经显示,所以它没有处理。

我想我可以将任务传递到对话框并让对话框启动它,但感觉这个问题有一个优雅的解决方案,不需要吗?

1 个答案:

答案 0 :(得分:3)

Shown事件非常适合解决此问题:

ProgressDialog dlg = new ProgressDialog();
dlg.Shown += (_, args) =>
{
    CreateTask().Start();
};
dlg.ShowDialog();

如果你需要在继续之前确保任务已经完成,并且表单可能在完成之前关闭,你可以修改它以等待这样的任务:

ProgressDialog dlg = new ProgressDialog();
Task longRunningTask = CreateTask();
dlg.Shown += (_, args) =>
{
    longRunningTask.Start();
};
dlg.ShowDialog();
longRunningTask.Wait();

请注意,在这两种情况下,都会在创建任务之前创建对话框表单,因此您可以将dlg实例传递给CreateTask,以便允许它访问表单的实例。 / p>