假设我有一个我想要运行的长时间运行的任务,并且可以通知用户界面其状态/进度。这里的问题是我想要的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时,我得到了可怕的跨线程异常,因为第一个事件在创建时到达进度对话框,但已经显示,所以它没有处理。
我想我可以将任务传递到对话框并让对话框启动它,但感觉这个问题有一个优雅的解决方案,不需要吗?
答案 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>