WinForm应用程序UI调用长耗时函数 z
需要在后台线程上执行长时间运行操作。
有几种方法可以做到这一点。
-
您可以将方法调用排队以在线程池线程上执行(请参阅here):
ThreadPool.QueueUserWorkItem(new WaitCallback(YourMethod));
在.NET 4.0中,您可以使用TaskFactory:
Task.Factory.StartNew(() => YourMethod());
在.NET 4.5及更高版本中,您可以(而且应该而不是
TaskFactory.StartNew()
)使用Task.Run()
:Task.Run(() => YourMethod());
-
如果您需要进度更新或完成后的通知,您可以使用BackgroundWorker来更好地控制方法。将BackgroundWorker控件拖到窗体上,并将方法附加到dowork事件。然后在您想要运行方法时启动工作程序。您当然可以从代码中手动创建BackgroundWorker,只需记住它在完成后需要处理。
-
为您的工作创建一个全新的线程。这是最复杂的,除非您需要对线程进行非常细粒度的控制,否则不是必需的。如果您想了解这一点,请参阅Thread课程的MSDN页面。
请记住,对于任何线程,您不能更新GUI,或从后台线程更改任何GUI控件。如果要在GUI上执行任何操作,则必须使用Invoke(和InvokeRequired)在GUI线程上触发该方法。请参阅here。
更新 UI 线程进度:
.NET 4.5 添加了 IProgress<T> 接口和 Progress<T> 类。内置的 Progress 类捕获同步上下文,就像 async/await 一样,然后当您报告进度时,它使用该上下文进行回调(在您的情况下为 UI 线程)。
private async void button1_Click(object sender, EventArgs e) { var progress = new Progress<ProcessViewModel>(UpdateRow); //报告进度时对 UpdateRow 进行回调. var tasks = Processes.Select(process => ProcessObject(process, progress)).ToList(); await Task.WhenAll(tasks); } private async Task ProcessObject(ProcessViewModel process, IProgress<ProcessViewModel> progress) { //循环 await Task.Run(()=> { //Will be run in ThreadPool thread //Do whatever cpu bound work here //还在线程池线程中 process.Progress++; // 反馈给 UI,在 UI 线程上调用 UpdateRow. progress.Report(process); }); }