C#跨线程访问控件,我们一般 ​​new​​ 一个线程,多线程来操作。今天本以为是这样操作的,可还是卡死程序了。记录一下碰到的这些疑难杂症。

一、怀疑多界面问题

其实,与嵌套多少个控件没关系。

C# 跨线程访问控件需要注意的问题_控件

▲ 我是在一个主窗体放了一个panel,然后,在panel里面加载menu1,menu1通过按钮加载menu2;再在么menu2里面new一个线程,跨线程访问textbox。可这样还是被卡死了。

怀疑是在主窗体嵌入的两个​​menu​​​,在​​menu​​里面这样操作不行。后面这种情况被我排除了。因为,我把代码复制到主页面,相同功能还是卡死。所以,还是代码有问题。

二、聚焦代码问题

问题其实是,​​this.textBox1.Invoke(/*调用代码*/)​​​的​​调用代码​不能太耗时,与控件本身操作无关的代码。不然就卡死

开始我是这样写的:

int a = 0;
thread = new Thread(() =>
{
if (this.textBox1.InvokeRequired) {
this.textBox1.Invoke(new Action(() => {
while (true) {
this.textBox1.Text = a++.ToString();
Thread.Sleep(1000);
}
}));
}
});

thread.IsBackground = true;
thread.Start();

把循环放到控件​​Invoke​​外面去

int a = 0;
thread = new Thread(() =>
{
if (this.textBox1.InvokeRequired) {
while (true) {
this.textBox1.Invoke(new Action(() => {
this.textBox1.Text = a++.ToString();
Thread.Sleep(1000);
}));
}
}
});

thread.IsBackground = true;
thread.Start();

上面这样还是不行的,​​Thread.Sleep(1000);​​​耗时,且与跨线程访问的控件无关。经测试即使换成耗时的循环​​for(int i = 0; i < 1000; i++) for(int j = 0; j < 100000; j++);​​,这样照样会卡死。

把​​Thread.Sleep(1000);​​​放​​Invoke​​外面去

这样能够工作的,不卡。

int a = 0;
thread = new Thread(() =>
{
if (this.textBox1.InvokeRequired) {
while (true) {
this.textBox1.Invoke(new Action(() => {
this.textBox1.Text = a++.ToString();
}));
Thread.Sleep(1000);
}
}
});

thread.IsBackground = true;
thread.Start();

效果如下:

C# 跨线程访问控件需要注意的问题_加载_02

▲ 主页面的textbox计时显示和两层加载的menu中的textbox的计时显示也是没问题的,窗体不冻结卡死。

其实本质上为什么会这样,我是没有明确搞明白的,都是自己去理解,不确定是不是自己理解的这样。也不想花太多的时间去深究了,能实现效果就好。如何学会把时间花在最关键的地方?这是一门学问,越早明白越受益。