在开发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
方法应该被广泛使用。