Winform跨线程访问UI
阅读原文时间:2023年08月11日阅读:3

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