龙空技术网

Winform跨线程访问UI

小美短视频 505

前言:

此刻各位老铁们对“textbox invoke”可能比较关心,同学们都需要剖析一些“textbox invoke”的相关资讯。那么小编也在网摘上收集了一些关于“textbox invoke””的相关文章,希望各位老铁们能喜欢,看官们一起来了解一下吧!

在开发winfrom应用时,经常遇到异常:System.InvalidOperationException:“线程间操作无效: 从不是创建控件“xxxx”的线程访问它。出现这个异常的原因是创建这个UI的线程,和当前访问这个UI的线程不会是同一个。Winform为了防止线程不安全,因此对这个跨线程访问抛出异常,禁止这个操作。

解决方案

使用InvokeRequired属性判断是否线程安全。

            if (richTextBox1.InvokeRequired)

{

richTextBox1.Invoke(new Action(() =>

{

richTextBox1.AppendText(log);

richTextBox1.AppendText("\r\n");

}));

}

else

{

richTextBox1.AppendText(log);

richTextBox1.AppendText("\r\n");

}

如果richTextBox1是在非主线程创建或找不到其句柄,那么richTextBox1.InvokeRequired=false返回false,就会走else分支,如果在找不到句柄的情况下,else里的代码也会抛异常。为了更加安全,需要进一步对句柄进行判断,用IsHandleCreated判断是否创建了句柄。

            if (richTextBox1.InvokeRequired)

{

richTextBox1.Invoke(new Action(() =>

{

richTextBox1.AppendText(log);

richTextBox1.AppendText("\r\n");

}));

}

else

{

if (richTextBox1.IsHandleCreated)

{

richTextBox1.AppendText(log);

richTextBox1.AppendText("\r\n");

}

}

上面代码基本上没什么问题了。但是稍显麻烦,可以进行精简一下。使用哦当前FormInvoke方法而不是具体某个ControlInvoke,这样能确保当前的操作一定在当前的UI线程中,且句柄一并被创建。

        private void Log(string log)

{

Invoke(new Action(() =>

{

richTextBox1.AppendText(log);

richTextBox1.AppendText("\r\n");

}));

}

其实在winform中跨线程访问UI很常见,比如在一个子窗口中进行了某个操作,需要更新主窗口里的某些状态或数据,如果稍不注意就会出现跨线程访问UI的异常,因此Invoke方法应该被广泛使用。

标签: #textbox invoke