不安全的访问方式:
//获取或设置一个值,该值指示是否捕获对错误线程的调用,这些调用在调试应用程序时访问控件的 Handle 属性。
Control.CheckForIllegalCrossThreadCalls = false;
为什么说这种方式不安全?无需过多的演示,从字面上去理解:获取或设置一个值,该值指示是否捕获对错误线程的调用,这些调用在调试应用程序时访问控件的 Handle 属性(控件绑定到的窗口句柄)。
显而易见,将物品当着物主的面强行拿去用,还指不定时候还给他。
安全的访问方式:
这里提供两种解决方案,
1、委托
2、Action() + lambda表达式
例子:(这里将点击New Thread Visit 按钮 创建新线程,将新线程运行的结果显示在result set 按钮上)
效果:
方式1:(委托)
public partial class Form1 : Form
{
//创建委托
public delegate void _delegate(int count);
private _delegate my_delegate;
private Thread t1;
public Form1()
{
InitializeComponent();
//实例委托
my_delegate = show;
}
//New Thread
private void button1_Click(object sender, EventArgs e)
{
t1 = new Thread(one);
t1.Start();
}
//在button2中显示
private void show(int count) {
button2.Text = count.ToString();
}
private void one() {
int count = 0;
while (true)
{
count++;
Thread.Sleep(200);
//如果控件的 true 是在与调用线程不同的线程上创建的(说明您必须通过 Invoke 方法对控件进行调用),则为 Handle;否则为 false
if (button2.InvokeRequired)
{
//在拥有此控件的基础窗口句柄的线程上执行指定的委托
button2.Invoke(my_delegate,count);
}
}
}
}
方式二:(Action() + lambda表达式)
public partial class Form1 : Form
{
private Thread t1;
public Form1()
{
InitializeComponent();
}
//New Thread
private void button1_Click(object sender, EventArgs e)
{
t1 = new Thread(one);
t1.Start();
}
private void one() {
int count = 0;
while (true)
{
count++;
Thread.Sleep(200);
//如果控件的 true 是在与调用线程不同的线程上创建的(说明您必须通过 Invoke 方法对控件进行调用),则为 Handle;否则为 false
if (button2.InvokeRequired)
{
//在拥有此控件的基础窗口句柄的线程上执行指定的委托
button2.Invoke(
//封装一个方法,该方法不具有参数并且不返回值。
new Action(() =>{
button2.Text = count.ToString();
}));
}
}
}
}
xxx