C#跨线程操作控件的最简单实现探究

随着程序复杂度的提高,程序不可避免会出现多个线程,此时就很可能存在跨线程操作控件的问题。

跨线程操作UI控件主要有三类方式:

1、禁止系统的线程间操作检查。(此法不建议使用

2、使用Invoke(同步)或者BeginInvoke(异步)。(使用委托实现,并用lambda表达式简化代码

3、使用BackgroundWorker组件。(此法暂不介绍,详情可见文末的参考资料

先看一个跨线程操作失败的例子:

新建一个Winform窗口程序项目,拖一个button1和label1控件到Form1窗体上。启动程序以后试图通过点击button1改变label1的值,完整代码如下:

 1 using System;
 2 using System.Threading;
 3 using System.Windows.Forms;
 4
 5 namespace Windows跨线程调用控件
 6 {
 7     public partial class Form1 : Form
 8     {
 9         public Form1()
10         {
11             InitializeComponent();
12         }
13
14         private void button1_Click(object sender, EventArgs e)
15         {
16             Thread thread1 = new Thread(new ParameterizedThreadStart(UpdateLabel));
17             thread1.Start("label已更新");
18         }
19
20         private void UpdateLabel(object str)
21         {
22             label1.Text = str.ToString();
23         }
24     }
25 }

点击button1以后运行报错:

解决方案:

方法一:禁止系统的线程间操作检查。

代码就一句话:Control.CheckForIllegalCrossThreadCalls = false;通常写在Form1类的构造方法Form1()中。如下所示:

1 public Form1()
2         {
3             InitializeComponent();
4
5             Control.CheckForIllegalCrossThreadCalls = false;        
6         }

但是,这种方法是很不可靠的,有时候还是会报错。

方法二:使用Invoke(同步)或者BeginInvoke(异步)最精简的代码如下:

 1 using System;
 2 using System.Threading;
 3 using System.Windows.Forms;
 4
 5 namespace Windows跨线程调用控件
 6 {
 7     public partial class Form1 : Form
 8     {
 9         public Form1()
10         {
11             InitializeComponent();
12         }
13
14         private void button1_Click(object sender, EventArgs e)
15         {
16             Thread thread1 = new Thread(UpdateLabel);//可以省略线程的委托类型ParameterizedThreadStart
17             thread1.Start("label已更新");
18         }
19
20         private void UpdateLabel(object str)
21         {
22             if (label1.InvokeRequired)//不同线程为true,所以这里是true
23             {
24                 BeginInvoke(new Action<string> (x => {label1.Text = x.ToString();}),str);   
25             }
26         }
27     }
28 }

说明:Action是.NET预定义好的委托,可以简化委托的语法,如果不清楚它的用法,可以搜索“Action和Func的用法”。

将上面的两种方法总结在同一段程序里面如下所示:(注意看代码中的注释)

 1 using System;
 2 using System.Threading;
 3 using System.Windows.Forms;
 4
 5 namespace Windows跨线程调用控件
 6 {
 7     public partial class Form1 : Form
 8     {
 9         public Form1()
10         {
11             InitializeComponent();
12
13             /************* 方法一 ************/
14             //Control.CheckForIllegalCrossThreadCalls = false;
15
16         }
17
18         private void button1_Click(object sender, EventArgs e)
19         {
20             Thread thread1 = new Thread(new ParameterizedThreadStart(UpdateLabel));
21             thread1.Start("label已更新");
22         }
23
24
25         //如果控件的 Handle(句柄) 是在与调用线程不同的线程上创建的(说明您必须通过 Invoke 方法对控件进行调用),则为 true;否则为 false。
26         private void UpdateLabel(object str)
27         {
28             if (label1.InvokeRequired)//当是不同的线程在访问时为true,所以这里是true
29             {
30                 /************* 方法二 ************/
31                 //Action<string> actionDelegate = (x) => { this.label1.Text = x.ToString(); };
32
33                 ////如果已经创建控件的句柄,则除了 InvokeRequired 属性以外,控件上还有四个可以从【任何线程】上安全调用的方法,
34                 ////它们是:Invoke、BeginInvoke、EndInvoke 和 CreateGraphics。
35                 ////在后台线程上创建控件的句柄之前调用 CreateGraphics 可能会导致非法的跨线程调用。
36                 ////对于所有其他方法调用,则应使用调用 (invoke) 方法之一封送对控件的线程的调用。
37                 //this.label1.BeginInvoke(actionDelegate, str);
38
39
40                 /************* 方法二(变式) ************/
41                 //也可以直接用下面一句话来完成
42                 //Control.BeginInvoke 方法有两个重载:BeginInvoke(Delegate)    ,BeginInvoke(Delegate, Object[]),下式用的是第二个重载
43                 this.BeginInvoke(new Action<string>((x) => { label1.Text = x.ToString(); }), str);
44
45                 //如果启动的多线程不需要带可变的参数,那更简单:
46                 //label1.BeginInvoke(new Action(() => { label1.Text = "aaa"; }));
47             }
48         }
49     }
50 }

参考资料:

http://www.cnblogs.com/TankXiao/p/3348292.html

时间: 2024-10-13 12:07:53

C#跨线程操作控件的最简单实现探究的相关文章

C# 跨线程操作控件方法封装

Winform程序需要处理耗时操作时,往往需要将耗时操作放入新开的子线程进行处理,在子线程中可能会经常去修改或操作主线程上的控件: 如果直接在子线程中操作控件,会报线程间操作无效等错误,这里提供一个我自己经常使用的跨线程操作方式,代码如下: //跨线程操作控件 //(将数据全部装填完毕后,在一起放到主界面刷新控件:不要一边装填一边刷新主界面控件,这样依然会导致界面卡顿) public void showMsg(string str) { if (textBox.InvokeRequired) {

C#跨线程操作控件

1.首先通过按键创建子线程: 创建子线程,子线程调用changeText方法. 1 private void btnOK_Click(object sender, EventArgs e) 2 { 3 Thread th = new Thread(changeText); 4 th.Name = "new Thread!"; 5 th.IsBackground = true; 6 th.Start(); 7 } 2.子线程操作弹窗提示: 1 void changeText() 2 {

跨线程访问控件的问题和编程方法

很多时候写windows程序都需要结合多线程,经常会跨线程操作控件,这时就会出错,提示不允许"从不是创建控件的进程访问它". 这个时候的解决思路:把想对另一线程中的控件实施的操作放到一个函数中,然后使用delegate代理那个函数,并且在那个函数中加入一个判断,用InvokeRequired来判断调用这个函数的线程是否和控件线程在同一线程中,如果是则直接执行对控件的操作,否则利用控件的Invoke或BeginInvoke方法来执行这个代理,执行唤醒控件的操作.Invoke的中文解释是唤

2 跨线程访问控件InvokeHelper类

这里是一张动画,演示在多线程(无限循环+Thread.Sleep)情况下主界面操作不受影响. 多线程是一种提高程序运行效率和性能的常用技术.随着我们学习工作的深入,在编程中或多或少会涉及到需要多线程的情况.多数时候,我们的操作模式是后台线程中处理数据,计算结果,然后在前台界面(GUI)中更新显示. 在.NET Framework中,为了保证线程安全,避免出现访问竞争等问题,是不允许跨线程访问窗体控件的.如果强行访问,则会引发InvalidOperationException无效操作异常,如下图:

C# 跨线程调用控件

在C# 的应用程序开发中, 我们经常要把UI线程和工作线程分开,防止界面停止响应.  同时我们又需要在工作线程中更新UI界面上的控件, 下面介绍几种常用的方法 阅读目录 线程间操作无效 第一种办法:禁止编译器对跨线程访问做检查 第二种办法: 使用delegate和invoke来从其他线程中调用控件 第三种办法: 使用delegate和BeginInvoke来从其他线程中控制控件 第四种办法: 使用BackgroundWorker组件 源代码下载 线程间操作无效 界面上有一个button和一个la

多线程总结之旅(112):跨线程调用控件的几种方式

本来是写完线程池就结束多线程总结之旅系列的,但是想想平时在项目中用到线程仅仅不够的,为什么这么说呢?举个例子:我们有一个函数,它的功能就是加载数据,然后绑定到datagridview.现在我们开启一个线程去执行这个函数.结果可想而知,它会报错:提示线程无法访问...之类的话.为什么报错呢?因为你在开启的线程中操作了datagridview控件,也就是你跨线程调用控件了. 那么我们应该怎么跨线程调用控件呢?下面我就把我总结的几种方法奉献给各位. 跨线程调用控件的几种方法: 1.方法一:Contro

【转载】C# 跨线程调用控件

转自:http://www.cnblogs.com/TankXiao/p/3348292.html 感谢原作者,转载以备后用 在C# 的应用程序开发中, 我们经常要把UI线程和工作线程分开,防止界面停止响应.  同时我们又需要在工作线程中更新UI界面上的控件, 下面介绍几种常用的方法 阅读目录 线程间操作无效 第一种办法:禁止编译器对跨线程访问做检查 第二种办法: 使用delegate和invoke来从其他线程中调用控件 第三种办法: 使用delegate和BeginInvoke来从其他线程中控

WinForm跨线程访问控件异常

WinForm跨线程访问控件异常 最近做了个WinForm的小项目,遇到个简单的问题记录下.需求:点击"下载",显示正在下载,下载完后更新状态为"ready"(要求用多线程,避免前台卡死). 做法 新建一个线程,在新线程中下在更新控件状态,报异常跨线程程访问控件(当然可以通过关闭跨线程访问控件检验解决,不鼓励这么做).应该C#5.0版本添加的任务Task异步,这里用异步解决. Code 新建WinForm 项目,Form1添加控件如图 后台代码 private vo

Winform跨窗体操作控件(使用委托)

Winform跨窗体操作控件是winform开发中很常见的形式,最常见且简单有效的方式便是使用委托的方式来进行操作,下面我将通过一个小实例来说明如何使用委托跨窗体实现控件操作. 实例介绍:两个窗体,Form1按钮打开Form2,然后在Form2的TextBox输入值在Form1的TextBox中显示出来. 一.项目结构 Form1窗体设计: Form2窗体设计: 二.代码实现 在Form2的代码中声明一个委托(写在Form2类外)-> public delegate void ChangeFor