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

本文关键字:任务 模式 对话框 TPL | 更新日期: 2023-09-27 17:58:04

假设我有一个长时间运行的Task要运行,它可以通知用户界面它的状态/进度。这里的问题是,我想要的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时,我会引发可怕的跨线程异常,因为第一个事件在创建进度对话框时到达进度对话框,但尚未显示,因此没有句柄。

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

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

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