在C#中,从另一个线程更新GUI通常需要使用UI线程的同步上下文(SynchronizationContext
)或者专门用于在UI线程执行操作的方法(Control.Invoke
或Dispatcher.Invoke
)。下面分别给出在WinForms和WPF中的基础功能和高级功能的实例源代码。
创建WinForms应用程序: 创建一个包含按钮和标签的WinForms应用程序。
创建后台线程: 使用Thread
类或Task.Run
创建一个后台线程,模拟耗时操作。
使用Invoke更新UI: 在后台线程中使用Control.Invoke
更新UI。
using System;
using System.Threading;
using System.Windows.Forms;
namespace WinFormsThreadedGUIUpdate
{
public partial class MainForm : Form
{
public MainForm()
{
InitializeComponent();
}
private void btnStartTask_Click(object sender, EventArgs e)
{
// 启动后台任务
Thread thread = new Thread(PerformTask);
thread.Start();
}
private void PerformTask()
{
// 模拟耗时操作
Thread.Sleep(3000);
// 在UI线程上更新标签文本
UpdateLabel("Task 完成!");
}
private void UpdateLabel(string text)
{
if (lblStatus.InvokeRequired)
{
// 在UI线程上异步执行更新操作
lblStatus.Invoke(new Action(() => lblStatus.Text = text));
}
else
{
// 在UI线程上直接更新操作
lblStatus.Text = text;
}
}
}
}
创建WPF应用程序: 创建一个包含按钮和标签的WPF应用程序。
创建后台线程: 使用Task.Run
创建一个后台线程,模拟耗时操作。
使用Dispatcher更新UI: 在后台线程中使用Dispatcher.Invoke
更新UI。
using System;
using System.Threading;
using System.Windows;
namespace WPFThreadedGUIUpdate
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
private void btnStartTask_Click(object sender, RoutedEventArgs e)
{
// 启动后台任务
Task.Run(() =>
{
// 模拟耗时操作
Thread.Sleep(3000);
// 在UI线程上更新标签文本
UpdateLabel("Task 完成!");
});
}
private void UpdateLabel(string text)
{
if (Dispatcher.CheckAccess())
{
// 在UI线程上直接更新操作
lblStatus.Content = text;
}
else
{
// 在UI线程上异步执行更新操作
Dispatcher.Invoke(new Action(() => lblStatus.Content = text));
}
}
}
}
在高级功能中,WinForms和WPF均可以使用SynchronizationContext
进行线程间通信,代码更为简洁。
using System;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace AdvancedThreadedGUIUpdate
{
public partial class MainForm : Form
{
private readonly SynchronizationContext synchronizationContext;
public MainForm()
{
InitializeComponent();
// 获取当前同步上下文
synchronizationContext = SynchronizationContext.Current;
}
private void btnStartTask_Click(object sender, EventArgs e)
{
// 启动后台任务
Task.Run(() =>
{
// 模拟耗时操作
Thread.Sleep(3000);
// 在UI线程上更新标签文本
synchronizationContext.Post(new SendOrPostCallback(UpdateLabel), "Task 完成!");
});
}
private void UpdateLabel(object state)
{
// 在UI线程上更新标签文本
lblStatus.Text = state.ToString();
}
}
}
在WPF中,Dispatcher
本身就是一种同步上下文,所以代码更为简洁,无需额外的同步上下文对象。高级功能的WPF示例代码与基础功能示例代码相似。
源代码获取:公众号回复消息【code:12166
】