前言:
此刻各位老铁们对“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");
}
}
上面代码基本上没什么问题了。但是稍显麻烦,可以进行精简一下。使用哦当前Form的Invoke方法而不是具体某个Control的Invoke,这样能确保当前的操作一定在当前的UI线程中,且句柄一并被创建。
private void Log(string log)
{
Invoke(new Action(() =>
{
richTextBox1.AppendText(log);
richTextBox1.AppendText("\r\n");
}));
}
其实在winform中跨线程访问UI很常见,比如在一个子窗口中进行了某个操作,需要更新主窗口里的某些状态或数据,如果稍不注意就会出现跨线程访问UI的异常,因此Invoke方法应该被广泛使用。
标签: #textbox invoke