C# 线程调用主线程中的控件

Posted 寒夜美美

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C# 线程调用主线程中的控件相关的知识,希望对你有一定的参考价值。

由于项目的需要,最近几天一直在做串口和数据库。由于C#使用的时间不长,所以在编写代码和调试的过程中总是遇到意想不到的问题,比如在使用串口接收数据的时候,在接收数据事件中想把接收的数据放入一个textbox作显示,但是明明非常简单的代码,在编译的时候总是提示有错误。后来查看网上资料,才知道C#还有委托,匿名等等之类的新东西。下面我就把我这几天的经验和大家分享一下。这次就主要说说委托和匿名方法,以后在说说串口使用方面的经验。

先说一下委托的基本概念,委托是一种引用型的数据类型,其实它的概念和C语言的函数指针几乎是一样的。回忆一下C语言的函数指针,定义一个函数指针,需要指定形参的类型和返回值的类型,只要有函数的形参类型和返回类型和这个函数指针一致,那么该函数指针就可以指向这个函数。C语言学习中一定会提高一个四则运算的例子,就是利用函数指针。

C#中委托的申明如下

【访问修饰符】 delegate 返回值类型委托名(【参数列表】);

委托虽然是一种数据格式,但是却需要像类一下去实例化。所以委托的实例化如下

委托类型委托变量名 = new 委托型构造函数(委托要引用的方法名)

实例化过后就可以使用命名方法和匿名方法两种方法,来来指定这个委托的函数引用。

废话少说,还是来一个实例吧。比如说,在主线程之外开辟一个线程,这个线程使用主线程的一个label控件,显示当前的时间。由于新开的线程要使用其他线程开的控件,所以需要使用到control类型的Invoke方法,而这个方法传入正是delegate类型。

先来看一个“想当然”的代码。

private void showDateTimeMethod()

{

while (true)

{

    //显示当前时间

label1.Text = "当前时间 " + DateTime.Now.ToString();

//线程暂停

Thread.Sleep(1000);

}

}

private void Form1_Load(object sender, EventArgs e)

{

//新建一个线程

Thread showDateTimethread = new Thread(new ThreadStart(showDateTimeMethod));

//该线程为后台线程

showDateTimethread.IsBackground = true;

//线程启动

showDateTimethread.Start();

}

       用我的大腿想想,上面的代码应该是对的,但是由于net苛刻的安全机制,showDateTimeMethod中的代码是有错误的。为了解决这个错误可以声明一个委托类型,实例化一个委托变量(这话听起来很拗口),并给这个委托变量指定函数引用。

       方法一,使用命名方法

    //声明一个委托类型,该委托类型无输入参数和输出参数

public delegate void ProcessDelegate();  

//函数引用,label控件显示当前时间,输入参数无,输出参数无,和声明的委托类型形式一致

public void LabelShow()

    {

        label1.Text = "当前时间 " + DateTime.Now.ToString();

}

然后在线程中实例化一个委托变量,指向这个函数引用。

while (true)

{

//使用命名方法

ProcessDelegate showProcess = new ProcessDelegate(LabelShow);

//调用label的invoke方法

    label1.Invoke(showProcess);

//线程暂停

Thread.Sleep(1000);

}

这样的话就可以在窗体中看到当前的时间了。

方法二,使用匿名方法

刚刚的函数引用只有区区一行,这一行完全可以使用匿名方法实现,如果使用匿名方法的话就可以免去编写一个函数,代码显得非常简洁。

匿名方法使用方法也很简单。

委托类型委托变量 = delegate (【参数列表】){代码块};

while (true)

{

//实例化一个委托变量,使用匿名方法构造

ProcessDelegate showProcess = delegate()

        {

           label1.Text = "当前时间 " + DateTime.Now.ToString();

        };

label1.Invoke(showProcess);

//线程暂停

Thread.Sleep(1000);

}

方法三,使用MethodInvoker

再观察一下上面的代码,这个委托类型,输入参数无输出参数也无,就可以使用C#的MethodInvoker直接构造,关于MethodInvoker,MSDN给出这样一个解释“表示一个委托,该委托可执行托管代码中声明为 void 且不接受任何参数的任何方法”

//使用匿名方法2

while (true)

{

        label1.Invoke

       (

       //委托,托管无参数的任何方法

           new MethodInvoker

           (

               delegate

              {

                  label1.Text = "当前时间 " + DateTime.Now.ToString();

               }

           )

);

//线程暂停

Thread.Sleep(1000);

}

 

如果熟悉了C#的委托和匿名方法之后,那么在使用串口控件的时候就方便了很多。但是解决了旧的问题,新的问题也会到来。如果串口获得的数据很多,直接在接收的时候处理的话很难保证实时性,代码也变得很臃肿不堪。通过查阅网上的资料,想尝试使用异步的方法处理问题,即串口接收事件只负责接收数据,并把数据放入队列中一个线程每隔一段时间处理线程的内容,然后把整理好的数据放入数据库中。不过这要下次再说了啊。

 

出自:http://blog.csdn.net/xukai871105/article/details/6204563

以上是关于C# 线程调用主线程中的控件的主要内容,如果未能解决你的问题,请参考以下文章

C#如何回到主线程,如何在委托指定线程执行

c# 多线程运行时出现主界面卡死现象如何解决?

C# 使用代理实现线程间调用

C# winform 跨线程更改窗体控件的属性

c# 定时器调用控件的事件处理事情出现卡死主界面如何解决

wpf中 我新开一个线程添加控件到主窗体