C# WPF中的GUI多线程技巧详解

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模式使得异步编程更加简单。开发者应根据具体的应用需求和场景,选择最合适的多线程实现方式。

往期精品推荐:

在国内默默无闻的.NET,在国外火的超乎想象?

C#的膨胀之路:创新还是灭亡

介绍.NET 6款好看的winform开源UI库

介绍一款最受欢迎的.NET 开源UI库

WPF第三方开源UI框架:打造独特体验的魔法师

WPF与Winform,你的选择是?

WinForm的前世今生

.NET成年了,然后呢?------编程界的逆袭传奇

相关推荐
yqcoder4 分钟前
reactflow 中 useNodesState 模块作用
开发语言·前端·javascript
Heaphaestus,RC6 分钟前
【Unity3D】获取 GameObject 的完整层级结构
unity·c#
baivfhpwxf202315 分钟前
C# 5000 转16进制 字节(激光器串口通讯生成指定格式命令)
开发语言·c#
许嵩6618 分钟前
IC脚本之perl
开发语言·perl
长亭外的少年29 分钟前
Kotlin 编译失败问题及解决方案:从守护进程到 Gradle 配置
android·开发语言·kotlin
直裾29 分钟前
Scala全文单词统计
开发语言·c#·scala
心仪悦悦30 分钟前
Scala中的集合复习(1)
开发语言·后端·scala
JIAY_WX32 分钟前
kotlin
开发语言·kotlin
代码小鑫1 小时前
A043-基于Spring Boot的秒杀系统设计与实现
java·开发语言·数据库·spring boot·后端·spring·毕业设计