当我尝试访问WinForms控件时,我收到错误Control accessed from a thread other than the thread it was created on
。我知道控制的所有修改都应该在UI线程中执行(需要BeginInvoke()
等),但我需要我的控件只能读取。
这是我的简化代码:
string text = textBox.Text;
从另一个线程访问WinForms控件的属性值的模式是什么?
答案 0 :(得分:5)
对于像这样简单的东西,你不必专门使用BeginInvoke,你也可以使用Invoke,但是你确实需要在UI线程上调用调用。您可以使用一些魔法来隐藏几个方法调用中令人讨厌的细节,然后使用扩展方法使其更清晰。例如,假设我想用一些用于获取和设置Text属性的安全方法来扩展TextBox控件。我可能会这样做:
namespace System.Windows.Forms
{
public static class TextBoxExtensions
{
public static string GetTextThreadSafe(this TextBox box)
{
return GetTextBoxText(box);
}
public static void SetTextThreadSafe(this TextBox box, string str)
{
SetTextBoxText(box, str);
}
public static string GetTextBoxText(TextBox box)
{
if (box.InvokeRequired)
{
Func<TextBox, string> deleg = new Func<TextBox, string>(GetTextBoxText);
return box.Invoke(deleg, new object[] { box }).ToString();
}
else
{
return box.Text;
}
}
public static void SetTextBoxText(TextBox box, string str)
{
if (box.InvokeRequired)
{
Action<TextBox, string> deleg = new Action<TextBox, string>(SetTextBoxText);
box.Invoke(deleg, new object[] { box, str });
}
else
{
box.Text = str;
}
}
}
}
然后在另一个线程中你可以像这样调用文本框:
Thread t = new Thread(new ThreadStart(() =>
{
// Threadsafe call to set the text
SomeTextBox.SetTextThreadSafe("asdf");
// Threadsafe call to get the text
MessageBox.Show(SomeTextBox.GetTextThreadSafe());
}));
t.IsBackground = true;
t.Start();
答案 1 :(得分:0)
您必须使用BeginInvoke。如果需要返回值(例如,控件的Text内容),可以使用EndInvoke等待完成。
那就是说,你可能想要考虑做其他事情;让GUI线程“推送”数据到后台工作线程。这有助于减少与用户输入竞争的机会,并且可以实现更清晰的设计,并清晰地分离GUI和核心逻辑。