Winform跨线程访问UI

  • Winform跨线程访问UI已关闭评论
  • 129 次浏览
  • A+
所属分类:.NET技术
摘要

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

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

解决方案

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

            if (richTextBox1.InvokeRequired)             {                 richTextBox1.Invoke(new Action(() =>                {                    richTextBox1.AppendText(log);                    richTextBox1.AppendText("rn");                }));             }             else             {                 richTextBox1.AppendText(log);                 richTextBox1.AppendText("rn");             } 

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

            if (richTextBox1.InvokeRequired)             {                 richTextBox1.Invoke(new Action(() =>                {                    richTextBox1.AppendText(log);                    richTextBox1.AppendText("rn");                }));             }             else             {                 if (richTextBox1.IsHandleCreated)                 {                     richTextBox1.AppendText(log);                     richTextBox1.AppendText("rn");                 }              } 

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

        private void Log(string log)         {             Invoke(new Action(() =>             {                 richTextBox1.AppendText(log);                 richTextBox1.AppendText("rn");             }));         } 

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