在多线程开发过程中,有时候需要更新UI控件内容,但是在c#多线程Task、Thread、BackgroundWork中不能直接更新UI控件,否则会报调用线程不能访问此对象,因为它由另一个线程拥有The calling thread cannot access this object because a different thread owns it.异常,这个时候我们就需要用到控件委托 Invoke、 BeginInvoke、 EndInvoke来避免此异常;在winform和wpf多线程操作UI控件都会有这个问题,我们先讲讲winform下如何解决这个问题。
WINFORM
首先我们创建一个winform窗体,点击按钮时循环一万次并更新下方的richTextBox1内容,这个时候我们不加Invoke,运行代码后出现报错线程间操作无效:从不是创建控件“richTextBox1”的线程访问它。
这个时候我们加上Invoke委托,问题得到解决,分别贴上Invoke、 BeginInvoke用法,示例代码如下~
Invoke示例代码
public partial class Form2 : Form
{
public Form2()
{
InitializeComponent();
}
private void button1_Click(object sender, EventArgs e)
{
Task.Run(() =>
{
for (int i = 0; i < 10000; i++)
{
this.Invoke(new MethodInvoker(() =>
{
richTextBox1.AppendText($"第{i}次循环\r\n");
}));
}
});
}
}
BeginInvoke示例代码
public partial class Form2 : Form
{
public Form2()
{
InitializeComponent();
}
private void button1_Click(object sender, EventArgs e)
{
Task.Run(() =>
{
for (int i = 0; i < 10000; i++)
{
this.richTextBox1.BeginInvoke(new MethodInvoker(() =>
{
richTextBox1.AppendText($"第{i}次循环\r\n");
}));
}
});
}
}
WPF
在wpf中Invoke的位置和winform稍有不同,wpf Invoke是在控件的Dispatcher对象下,分别贴上Invoke、 BeginInvoke用法,示例代码如下~
Invoke示例代码
private void button1_Click(object sender, RoutedEventArgs e)
{
Task.Run(() =>
{
for (int i = 0; i < 10000; i++)
{
this.richTextBox1.Dispatcher.Invoke((Action)delegate
{
richTextBox1.AppendText($"第{i}次循环\r\n");
});
}
});
}
BeginInvoke示例代码
private void button1_Click(object sender, RoutedEventArgs e)
{
Task.Run(() =>
{
for (int i = 0; i < 10000; i++)
{
this.Dispatcher.BeginInvoke(System.Windows.Threading.DispatcherPriority.Normal, (Action)delegate
{
richTextBox1.AppendText($"第{i}次循环\r\n");
});
}
});
}