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时,我会引发可怕的跨线程异常,因为第一个事件在创建进度对话框时到达进度对话框,但尚未显示,因此没有句柄。
我想我可以将任务传递到对话框中,并让对话框启动它,但感觉有一个优雅的解决方案可以解决这个问题,而不需要这样做吗?
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
。