一、问题总结

 1. 在WinForm开发过程中用到线程时,往往需要在线程中访问线程外的控件,比如:设置textbox的Text值等等。如果直接访问UI控件会报出“从不是创建控件的线程访问它”错误。控件是在主线程中创建的(比如this.Controls.Add(...);),在其它线程直接访问主线程控件,与主线程发生线程冲突。

解决方法:

在控件响应函数中调用控件的Invoke方法,Invoke方法会顺着控件树向上搜索,直到找到创建控件的那个线程(通常是主线程),然后进入那个线程改变控件的外观,确保不发生线程冲突。

MSDN中说:
获取一个值,该值指示调用方在对控件进行方法调用时是否必须调用 Invoke 方法,因为调用方位于创建控件所在的线程以外的线程中。
如果控件的 Handle 是在与调用线程不同的线程上创建的(说明您必须通过 Invoke 方法对控件进行调用),则为 true;否则为 false。
Windows 窗体中的控件被绑定到特定的线程,不具备线程安全性 。因此,如果从另一个线程调用控件的方法,那么必须使用控件的一个 Invoke 方法来将调用封送到适当的线程。该属性可用于确定是否必须调用 Invoke 方法,当不知道什么线程拥有控件时这很有用。   

2. invoke表示同步,begininvoke表示异步。

3.在使用invoke前,判断IsHandleCreated 此属性指示控件是否有与他关联的句柄,如果已经为控件分配了句柄,则为 true;否则为 false。

 加此判断可以避免在退出程序的时候,如果此时调用了invoke中的控件,就会出现错误“在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke。”

二、代码举例

 1.线程访问主线程控件  

using System;
using System.Windows.Forms;
using System.Threading;
namespace WindowsFormsApp1
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }
        private void Form1_Load(object sender, EventArgs e)
        {

        }
        private void btnStart_Click(object sender, EventArgs e)
        {
            Thread th = new Thread(new ThreadStart(DoWork));
            th.Start();         

        }     
        private void DoWork()
        {          
            ShowMsg("hello world");           
        }

        delegate void ShowMsgDelegate(string msg); 
        //普通委托
        private void ShowMsg(string msg)
        {
            if(lblMsg.InvokeRequired)
            {
                ShowMsgDelegate showMsgDelegate = new ShowMsgDelegate(ShowMsg);
                lblMsg.Invoke(showMsgDelegate, new object[] { msg });
            }
            else
            {
                lblMsg.Text = msg;
            }           

        }

        //匿名代理
        private void ShowMsg2(string msg)
        {
            ShowMsgDelegate showMsgDelegate = delegate (string str)
            {
                lblMsg.Text = str;
            };
            lblMsg.Invoke(showMsgDelegate, new object[] { msg });

        }

        //在C# 3.0及以后的版本中有了Lamda表达式,
        //像上面这种匿名委托有了更简洁的写法。
        private void ShowMsg3(string msg)
        {
            this.Invoke(new Action(() => {
                lblMsg.Text = msg;
            }
            ));
        }

       
    }
}

 

private void ShowMsg(string msg, bool isOk)
{
    if (this.IsHandleCreated)
    {
        this.Invoke(new Action(() =>
        {
            lblMsg.Text = msg;
            lblMsg.ForeColor = isOk ? Color.Green : Color.Red;

        }));
    }
}

 

 

 

lambda写法

  private void button1_Click(object sender, EventArgs e)
        {
            Console.WriteLine("start");
            testThread =new Thread(() =>{

                for(int i=0;i<20;i++)
                {
                    Console.WriteLine(i);
                }
               
            });
            testThread.Start();
            Console.WriteLine("end");
        }

执行结果:

start
end
0
1
2
3
4
5
6
7
8
9