1. 使用BackgroundWorker组件
代码示例:
go
public partial class MainWindow : Window
{
private BackgroundWorker backgroundWorker = new BackgroundWorker();
public MainWindow()
{
InitializeComponent();
backgroundWorker.DoWork += BackgroundWorker_DoWork;
backgroundWorker.RunWorkerCompleted += BackgroundWorker_RunWorkerCompleted;
backgroundWorker.WorkerReportsProgress = true;
backgroundWorker.WorkerSupportsCancellation = true;
}
private void StartButton_Click(object sender, RoutedEventArgs e)
{
backgroundWorker.RunWorkerAsync();
}
private void BackgroundWorker_DoWork(object sender, DoWorkEventArgs e)
{
// 长时间运行的任务
for (int i = 0; i < 100; i++)
{
if (backgroundWorker.CancellationPending)
{
e.Cancel = true;
return;
}
backgroundWorker.ReportProgress(i + 1);
Thread.Sleep(100);
}
}
private void BackgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
if (e.Cancelled)
{
MessageBox.Show("Task was canceled.");
}
else if (e.Error != null)
{
MessageBox.Show("Error: " + e.Error.Message);
}
else
{
MessageBox.Show("Task completed.");
}
}
}
优点:
-
提供了简单的方式来异步执行操作。
-
支持进度更新和取消操作。
缺点:
- 编程模型较为古老,不是基于任务并行库(TPL)。
使用场景:
- 当需要简单的异步操作且不需要复杂的任务调度时。
2. 使用Task并行库(TPL)
代码示例:
go
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
private async void StartButton_Click(object sender, RoutedEventArgs e)
{
await Task.Run(() =>
{
// 长时间运行的任务
for (int i = 0; i < 100; i++)
{
Application.Current.Dispatcher.Invoke(() =>
{
// 更新UI
ProgressTextBlock.Text = $"Progress: {i + 1}";
});
Thread.Sleep(100);
}
});
ProgressTextBlock.Text = "Task completed.";
}
}
优点:
-
基于.NET的Task并行库,是现代异步编程的标准。
-
支持任务取消、继续与异常处理。
缺点:
- 需要正确处理UI线程和后台线程之间的上下文切换。
使用场景:
- 当需要进行复杂的异步编程和任务调度时。
3. 使用Dispatcher进行线程间通信
代码示例:
go
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
private void StartButton_Click(object sender, RoutedEventArgs e)
{
Thread backgroundThread = new Thread(DoWork);
backgroundThread.Start();
}
private void DoWork()
{
for (int i = 0; i < 100; i++)
{
Application.Current.Dispatcher.Invoke(() =>
{
// 更新UI
ProgressTextBlock.Text = $"Progress: {i + 1}";
});
Thread.Sleep(100);
}
Application.Current.Dispatcher.Invoke(() =>
{
ProgressTextBlock.Text = "Task completed.";
});
}
}
优点:
-
允许直接控制线程的创建和管理。
-
可以精确控制UI更新。
缺点:
- 需要手动管理线程生命周期,增加了复杂性。
使用场景:
- 当需要直接控制后台线程的行为时。
4. 使用数据绑定和INotifyPropertyChanged接口
代码示例:
go
public class ViewModel : INotifyPropertyChanged
{
private string _progress;
public string Progress
{
get { return _progress; }
set
{
if (_progress != value)
{
_progress = value;
OnPropertyChanged(nameof(Progress));
}
}
}
public event PropertyChangedEventHandler PropertyChanged;
protected virtual void OnPropertyChanged(string propertyName)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
}
public void DoWork()
{
// 长时间运行的任务
for (int i = 0; i < 100; i++)
{
Thread.Sleep(100);
Progress = $"Progress: {i + 1}";
}
}
}
优点:
-
通过数据绑定自动更新UI,减少了代码量。
-
遵循MVVM设计模式,提高了代码的可维护性。
缺点:
- 需要实现INotifyPropertyChanged接口,增加了实现的复杂性。
使用场景:
- 当应用程序遵循MVVM设计模式时。
5. 使用Async/Await模式
代码示例:
go
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
private async void StartButton_Click(object sender, RoutedEventArgs e)
{
await DoWorkAsync();
}
private async Task DoWorkAsync()
{
for (int i = 0; i < 100; i++)
{
await Task.Delay(100);
ProgressTextBlock.Text = $"Progress: {i + 1}";
}
ProgressTextBlock.Text = "Task completed.";
}
}
优点:
-
代码简洁,易于理解和维护。
-
自动处理线程间上下文切换。
缺点:
- 不适用于所有类型的长时间运行任务。
使用场景:
- 当需要在WPF应用程序中执行异步操作时。
总结
在C# WPF应用程序中,合理使用多线程技术可以显著提高应用程序的性能和用户体验。BackgroundWorker组件提供了一种简单的方式来执行后台操作;Task并行库(TPL)是现代异步编程的标准;Dispatcher是WPF中进行线程间通信的关键;数据绑定和INotifyPropertyChanged接口可以自动更新UI;Async/Await模式使得异步编程更加简单。开发者应根据具体的应用需求和场景,选择最合适的多线程实现方式。
往期精品推荐: