Winform中如何跨线程访问UI元素

在C# 的应用程序开发中, 我们经常要把UI线程和工作线程分开,防止界面停止响应, 同时我们又需要在工作线程中更新UI界面上的控件。但直接访问会出现“线程间操作无效”的情况,因为.NET禁止了跨线程调用控件, 否则谁都可以操作控件,最后可能造成错误。 下面介绍几种跨线程访问的方法:

1、禁止对跨线程访问做检查 (不推荐使用这种方法)

这种方法不检查跨线程访问,允许各个线程操作UI元素,容易出现错误。

public Form2()
{
  InitializeComponent();
  //禁止对跨线程访问做检查 (不推荐使用这种方法)
  Control.CheckForIllegalCrossThreadCalls = false;
}

2、使用委托方法  将其委托给UI控件更新

//使用委托方法 将其委托给UI控件更新
private void button1_Click(object sender, EventArgs e)
{
  Thread thread1 = new Thread(new ParameterizedThreadStart(UpdateLabel2));
  thread1.Start("更新Label");
}

private void UpdateLabel2(object str)
{
  if (label2.InvokeRequired)
  {
    // 当一个控件的InvokeRequired属性值为真时,说明有一个创建它以外的线程想访问它
    Action<string> actionDelegate = (x) => { this.label2.Text = x.ToString(); };
    // 或者
    // Action<string> actionDelegate = delegate(string txt) { this.label2.Text = txt; };
    this.label2.Invoke(actionDelegate, str);
  }
  else
  {
    this.label2.Text = str.ToString();
  }
}

3、使用delegate和BeginInvoke来从其他线程中控制控件

只要把上面的 this.label2.Invoke(actionDelegate, str); 中的 Invoke 改为BeginInvoke方法就可以了。

Invoke方法和BeginInvoke方法的区别是:Invoke方法是同步的, 它会等待工作线程完成,BeginInvoke方法是异步的, 它会另起一个线程去完成工作线。

4、使用同步上下文:SynchronizationContext方法

该方法是取得主线程的上下文信息,然后在子线程将访问UI控件方法推送到UI上下文的消息队列里,使用POST或者Send;

private SynchronizationContext synchronizationContext;

private void button2_Click(object sender, EventArgs e)
{
  synchronizationContext = SynchronizationContext.Current;
  new Thread(() => { UpdateText("跨线程访问"); }).Start();
}
void UpdateText(string msg)
{
  synchronizationContext.Post(_ => this.label2.Text = msg, null);
}

5、使用BackgroundWorker组件(推荐使用这个方法)

BackgroundWorker是.NET里面用来执行多线程任务的控件,它允许编程者在一个单独的线程上执行一些操作。耗时的操作(如下载和数据库事务)。

public partial class FileManagerForm : Form
{
  FileInfo file ;
  BackgroundWorker bw;
  ServerFile server;
  public FileManagerForm(string filePath)
  {
    InitializeComponent();

    file = new FileInfo(filePath);

    long size = file.Length / 1024 / 1024;
    lblOrgSize.Text = (int)size+ "MB";
    bw = new BackgroundWorker();
    server = new ServerFile(file.Name);
  }

  private void FileManagerForm_Load(object sender, EventArgs e)
  {
    proUpFile.Minimum = 0;
    proUpFile.Maximum = 100;

    bw.WorkerReportsProgress = true;
    bw.WorkerSupportsCancellation = true;
    bw.DoWork += Bw_DoWork;
    bw.ProgressChanged += Bw_ProgressChanged;
    bw.RunWorkerCompleted += Bw_RunWorkerCompleted;
    bw.RunWorkerAsync();
  }
  private void Bw_DoWork(object sender, DoWorkEventArgs e)
  {
    using(FileStream fileRead= file.OpenRead())
    {
      long setp = file.Length / 100;
      while (file.Length > fileRead.Position)
      {
        if (bw.CancellationPending)
        {
          break;
        }

        byte[] bytes = new byte[1024];
        int count = fileRead.Read(bytes, 0, bytes.Length);

        long writeLength= server.UpFile(bytes, count);

        if(writeLength >proUpFile.Value* setp)
        {
          int size = (int)(writeLength / 1024 / 1024);
          bw.ReportProgress(proUpFile.Value + 1, size);
        }

      }
      server.Close();
    }
  }
  private void Bw_ProgressChanged(object sender, ProgressChangedEventArgs e)
  {
    proUpFile.Value= e.ProgressPercentage> proUpFile.Maximum?proUpFile.Maximum:e.ProgressPercentage;
    lblUpLoadSize.Text = e.UserState.ToString() + "MB";
  }

  private void Bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
  {
    if (this.proUpFile.Value == this.proUpFile.Maximum)
    {
      MessageBox.Show("文件发送成功!");
    }
    else
    {
      MessageBox.Show("文件发送失败!");
    }
    this.Close();
  }

  private void btnCancel_Click(object sender, EventArgs e)
  {
     bw.CancelAsync();
  }
}

以上就是Winform中如何跨线程访问UI元素的详细内容,更多关于Winform 访问UI元素的资料请关注我们其它相关文章!

(0)

相关推荐

  • visual studio 2019使用net core3.0创建winform无法使用窗体设计器

    微软发布正式版net core3.0后,迫不及待的想体验一下用visual studio 2019在net core3.0下创建winform程序.创建方法很简单,和以前visual studio版本步骤差不多. 创建完成之后,尴尬的事情发生了,无法使用窗体设计器,双击Form1.cs文件不行,使用快捷键shift+F7也不行,在网上找了很久,发现好多人都遇到过这种问题,目前有两种解决方案 方案1 项目中创建多目标框架,包含net framework和net core. 打开csproj文件,将

  • 浅谈C#跨线程调用窗体控件(比如TextBox)引发的线程安全问题

    如何:对 Windows 窗体控件进行线程安全调用 访问 Windows 窗体控件本质上不是线程安全的. 如果有两个或多个线程操作某一控件的状态,则可能会迫使该控件进入一种不一致的状态. 还可能会出现其他与线程相关的 Bug,例如争用情况和死锁. 确保以线程安全方式访问控件非常重要. 在未使用 Invoke 方法的情况下,从不是创建某个控件的线程的其他线程调用该控件是不安全的. 以下非线程安全的调用的示例. // This event handler creates a thread that

  • C#中跨线程访问控件问题解决方案分享

    net 原则上禁止跨线程访问控件,因为这样可能造成错误的发生,推荐的解决方法是采用代理,用代理方法来间接操作不是同一线程创建的控件. 第二种方法是禁止编译器对跨线程访问作检查,可以实现访问,但是出不出错不敢保证Control.CheckForIllegalCrossThreadCalls = false; 最近我在做一个项目,遇到了跨线程要去访问页面控件.但是总是提示出错,不能在其它线程中修改创建控件的线程的控件的值,后来采用了匿名代理,结果很轻松地解决了.解决过程如下:首先在窗体上,创建一个l

  • 在Winform程序中使用Spire.Pdf实现页面添加印章功能的实现

    在一些场合,我们往往需要使用印章来给每页文档加盖一个印章,以表示该文档经过某个部门的认证的,常规的做法就是打印文档后盖章,如果需要电子档再行扫描一下.这样的处理,如果文档很多,且仅仅需要电子文档的就很麻烦了,需要打印再盖章再扫描,而且电子文档往往有失真的问题.本篇随笔介绍如何在在Winform程序中使用Spire.Pdf实现页面添加印章处理,解决这一痛处. 1.在PDF增加印章处理 首先我们来了解下,用Adobe Acrobat Reader 进行文档的印章处理下,看看如何实现的,后面再用软件实

  • C#实现跨线程操作控件方法

    本文实例讲述了C#实现跨线程操作控件方法,分享给大家供大家参考.具体实现方法如下: 由于在.net平台下Winform.wpf禁止跨线程直接访问控件,因此在必须跨线程访问控件的时候采用异步方式. 1.winform项目中跨线程访问控件: 编写一个Winform小实例:在做winform项目中,有时为了将系统运行的状态实时显示到Form中,因此添加一个RichTextbox控件实时显示系统运行日志.本例实现的操作是将日志以字符串的形式写入RichTextbox控件,因为是实时显示,所以涉及到跨线程

  • Winform跨线程操作的简单方法

    分析:label标签控件是主线程创建的,不能直接从另一个线程访问.可以这样认为:不能跨线程直接访问控件; 最简单的办法就是: 复制代码 代码如下: using System.Windows.Forms; Parent.Invoke(new MethodInvoker(delegate {       Parent.label1.Text = "成功" ; })); 其中,"Parent"是主窗体名称. 这样,我们就是在需要窗体控件产生变化的时候,发消息通知主窗体改变

  • C# Winform调用百度接口实现人脸识别教程(附源码)

    百度是个好东西,这篇调用了百度的接口(当然大牛也可以自己写),人脸检测技术,所以使用的前提是有网的情况下.当然大家也可以去参考百度的文档. 话不多说,我们开始: 第一步,在百度创建你的人脸识别应用 打开百度AI开放平台链接: 点击跳转百度人脸检测链接,创建新应用 创建成功成功之后.进行第二步 第二步,使用API Key和Secret Key,获取 AssetToken 平台会分配给你相关凭证,拿到API Key和Secret Key,获取 AssetToken 接下来我们创建一个AccessTo

  • C#之WinForm跨线程访问控件实例

    本文实例讲述了C#中WinForm跨线程访问控件的实现方法,分享给大家供大家参考. 具体实现方法如下: 1.跨线程访问控件委托和类的定义 复制代码 代码如下: using System; using System.Windows.Forms; namespace ahwildlife.Utils {     /// <summary>     /// 跨线程访问控件的委托     /// </summary>     public delegate void InvokeDeleg

  • C# Winform中如何绘制动画示例详解

    前言 这里介绍一个.net自身携带的类ImageAnimator,这个类类似于控制动画的时间轴,使用ImageAnimator.CanAnimate可以判断一个图片是否为动画,调用ImageAnimator.Animate可以开始播放动画,即每经过一帧的时间触发一次OnFrameChanged委托,我们只要在该委托中将Image的活动帧选至下一帧再迫使界面重绘就可以实现动画效果了. 为了方便以后的使用,我将这些代码整合到了一起,形成一个AnimateImage类,该类提供了CanAnimate.

  • C#多线程与跨线程访问界面控件的方法

    本文实例讲述了C#多线程与跨线程访问界面控件的方法.分享给大家供大家参考.具体分析如下: 在编写WinForm访问WebService时,常会遇到因为网络延迟造成界面卡死的现象.启用新线程去访问WebService是一个可行的方法. 典型的,有下面的启动新线程示例: 复制代码 代码如下: private void LoadRemoteAppVersion()  {      if (FileName.Text.Trim() == "") return;      StatusLabel

随机推荐