WinForm中跨线程操作UI的解决方案汇总

 更新时间:2025年06月24日 09:31:18   作者:小码编匠  
在开发WinForm应用程序时,经常会遇到需要在线程间操作UI的情况,直接从非UI线程更新UI控件会导致异常,因此我们需要采取适当的方法来安全地进行这些操作,本文总结了几种常见的解决方法,并对其优缺点进行了分析,需要的朋友可以参考下

正文

方法一:禁用线程间的非法调用检查

这是最简单的方法,但也是最不推荐的做法。通过设置窗体属性Control.CheckForIllegalCrossThreadCalls = false;可以取消线程间的安全检查,从而允许跨线程更新UI。

然而,这种方法可能导致不稳定和不安全的行为,应避免使用。

public partial class one : Form
{
    public one()
    {
        InitializeComponent();
        Control.CheckForIllegalCrossThreadCalls = false;
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        Thread listen = new Thread(new ThreadStart(receive));
        listen.IsBackground = true;
        listen.Start();
    }
    private void receive()
    {
        UdpClient uc = new UdpClient(5839);
        while (true)
        {
            IPEndPoint ip = null;
            byte[] message = uc.Receive(ref ip);
            string messagestring = Encoding.UTF8.GetString(message);
            textBox1.Text = messagestring;
        }
    }
}

方法二:使用全局变量结合Timer实现

该方法利用了全局变量存储数据,并通过定时器(Timer)周期性地更新UI。虽然这种方法实现了目标,但由于它依赖于Timer的频率,可能会导致不必要的延迟和资源消耗。

public partial class two : Form
{
    string messagestring = "";

    public two()
    {
        InitializeComponent();
    }

    private void two_Load(object sender, EventArgs e)
    {
        Thread listen = new Thread(new ThreadStart(receive));
        listen.IsBackground = true;
        listen.Start();
        timer1.Start();
    }

    private void receive()
    {
        UdpClient uc = new UdpClient(5839);
        while(true)
        {
            IPEndPoint ip = null;
            byte[] message = uc.Receive(ref ip);
            messagestring = Encoding.UTF8.GetString(message);
        }
    }

    private void timer1_Tick(object sender, EventArgs e)
    {
        textBox1.Text = messagestring;
    }
}

方法三:使用BackgroundWorker组件

使用BackgroundWorker可以简化异步编程模型,适合处理简单的后台任务。

但是,它的局限性在于仅适用于Windows Forms,对于其他平台则不适用。

public partial class three : Form
{
    public three()
    {
        InitializeComponent();
    }

    private void backgroundWorker2_DoWork(object sender, DoWorkEventArgs e)
    {
        UdpClient uc = new UdpClient(5839);
        while(true)
        {
            IPEndPoint ip = null;
            byte[] message = uc.Receive(ref ip);
            string messagestring = Encoding.UTF8.GetString(message);
            backgroundWorker2.ReportProgress(50, messagestring);
        }
    }

    private void backgroundWorker2_ProgressChanged(object sender, ProgressChangedEventArgs e)
    {
        textBox1.Text = e.UserState.ToString();
    }
}

方法四:使用SynchronizationContext

通过SynchronizationContextPostSend方法可以在不同线程之间传递消息,确保UI更新安全执行。

这是一种较为灵活且可靠的方式。

public partial class fourth : Form
{
    SynchronizationContext SyncContext = null;

    public fourth()
    {
        InitializeComponent();
        SyncContext = SynchronizationContext.Current;
    }

    private void receive()
    {
        UdpClient uc = new UdpClient(5839);
        while (true)
        {
            IPEndPoint ip = null;
            byte[] message = uc.Receive(ref ip);
            string messagestring = Encoding.UTF8.GetString(message);
            SyncContext.Post(change,messagestring);
        }
    }

    private void change(object str)
    {
        textBox1.Text = str.ToString();
    }
}

方法五:使用Invoke或BeginInvoke

这是目前最常用的跨线程更新UI的方法。通过控件的InvokeBeginInvoke方法将委托调度到UI线程上执行,保证了线程安全性。

public partial class fifth : Form
{
    delegate void Change(string text);

    public fifth()
    {
        InitializeComponent();
    }

    private void Settext(string text)
    {
        textBox1.Text = text;
    }

    private void receive()
    {
        UdpClient uc = new UdpClient(5839);
        while (true)
        {
            IPEndPoint ip = null;
            byte[] message = uc.Receive(ref ip);
            string messagestring = Encoding.UTF8.GetString(message);
            this.BeginInvoke(new Change(Settext),messagestring);
        }
    }
}

总结

跨线程操作UI是WinForm开发中常见的挑战之一。尽管有多种方式可以解决这个问题,但考虑到安全性和灵活性,推荐使用SynchronizationContext或者控件的Invoke/BeginInvoke方法。正确理解并应用委托机制对有效管理多线程环境下的UI更新至关重要。

最后

到此这篇关于WinForm中跨线程操作UI的解决方案汇总的文章就介绍到这了,更多相关WinForm跨线程操作UI内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 基于C#实现文件伪装技术

    基于C#实现文件伪装技术

    这篇文章主要为大家详细介绍了如何基于C#实现文件伪装功能,将一般文件夹伪装成计算机,控制面板,打印机等,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-02-02
  • Winform圆形环绕的Loading动画实现代码

    Winform圆形环绕的Loading动画实现代码

    这篇文章主要介绍了Winform圆形环绕的Loading动画实现代码,有需要的朋友可以参考一下
    2014-01-01
  • 深入学习C#网络编程之HTTP应用编程(上)

    深入学习C#网络编程之HTTP应用编程(上)

    这篇文章主要介绍了如何学习C#网络编程之HTTP应用编程的相关知识,文中讲解的非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-06-06
  • unity置灰处理的实现

    unity置灰处理的实现

    本文主要介绍了unity置灰处理的实现,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧
    2021-07-07
  • C#函数式编程中的递归调用之尾递归详解

    C#函数式编程中的递归调用之尾递归详解

    这篇文章主要介绍了C#函数式编程中的递归调用详解,本文讲解了什么是尾递归、尾递归的多种方式、尾递归的代码实例等内容,需要的朋友可以参考下
    2015-01-01
  • C#打印日志的方法总结

    C#打印日志的方法总结

    在本篇文章里小编给大家整理了关于C#如何打印日志的技巧总结,需要的朋友们跟着学习下。
    2019-03-03
  • Unity幸运转盘实战项目

    Unity幸运转盘实战项目

    这篇文章主要为大家详细介绍了Unity幸运转盘实战项目,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-04-04
  • 深入解析C#编程中struct所定义的结构

    深入解析C#编程中struct所定义的结构

    这篇文章主要介绍了C#编程中struct所定义的结构,与C++一样,C#语言同时拥有类和结构,需要的朋友可以参考下
    2016-01-01
  • C#提取网页中超链接link和text部分的方法

    C#提取网页中超链接link和text部分的方法

    这篇文章主要介绍了C#提取网页中超链接link和text部分的方法,涉及C#正则表达式及字符串操作相关技巧,需要的朋友可以参考下
    2016-02-02
  • Unity AssetBundle打包工具示例详解

    Unity AssetBundle打包工具示例详解

    这篇文章主要介绍了Unity AssetBundle打包工具,本文通过实例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-10-10

最新评论