3
"Cross-thread operation is not valid" exception

我多次遇到此异常,但所有这些时间我都在设置控件的值。那时我使用一个名为 的函数解决了这个问题SetControlPropertyThreadSafe(),该函数仅由 stackoverflow.com 上的某个人建议。但是这次我在尝试获取组合框的值时遇到了这个异常。这是代码:

 string cat;
        private void button1_Click(object sender, EventArgs e)
        {
            if (textBox1.Text.Trim().Length > 20)
            {
                System.Threading.Thread t = new System.Threading.Thread(addQuickTopic);
                t.Start();
            }
            else
                MessageBox.Show("The length of the title must be greater than 20", "Title length invalid", MessageBoxButtons.OK, MessageBoxIcon.Exclamation);
        }
        public string tTitle="";
        void addQuickTopic()
        {
            Global.SetControlPropertyThreadSafe(button1, "Text", "Working...");
            Global.SetControlPropertyThreadSafe(button1, "Enabled", false);
            Topic t = new Topic();
            t.setTitle(textBox1.Text);
            t.setDescription(" ");
            t.setDesID(Global.getMd5Hash(Common.uid+DateTime.Today.ToString()+DateTime.Today.Millisecond.ToString()));
            t.setUsrID(Common.uid);
            t.setReplyToID("");
            t.setConDate("0");
            cat = CategoryList.SelectedValue.ToString();

如您所见,我直接获取 textBox1.Text 而不应用任何线程安全操作。但是在最后一行尝试获取组合框的选定值时,我得到了这个异常。那么有人可以建议我在这种情况下该怎么做吗?以下是用于设置控件值的线程安全函数的代码:

public static void SetControlPropertyThreadSafe(Control control, string propertyName, object propertyValue)
        {
            try
            {
                if (control.InvokeRequired)
                {
                    control.Invoke(new SetControlPropertyThreadSafeDelegate(SetControlPropertyThreadSafe), new object[] { control, propertyName, propertyValue });
                }
                else
                {
                    control.GetType().InvokeMember(propertyName, BindingFlags.SetProperty, null, control, new object[] { propertyValue });
                }
            }
            catch (Exception)
            { }
        }

我是否需要创建类似的get功能?或者任何其他可用的解决方案?

4

2 回答 2

4

现在,您从TextBox, via:获得价值t.setTitle(textBox1.Text);。这也会失败。

我是否需要创建类似的 get 函数?或者任何其他可用的解决方案?

是的,您需要一个 get 选项。但是,我建议不要使用反射和文本,并将其重新设计为使用通用方法和 lambda。

public static void SetControlPropertyThreadSafe<T>(T control, Action<T> action) where T : Control
    {
            if (control.InvokeRequired)
            {
                control.Invoke(action);
            }
            else
            {
                action();
            }
    }

这将允许您编写这个强类型,即:

Global.SetControlPropertyThreadSafe(button1, b => b.Text = "Working...");

您还可以创建一个强类型的 get 方法:

public static U GetControlPropertyThreadSafe<T,U>(T control, Func<T,U> func) where T : Control
{
     if (control.InvokeRequired)
     {
         return (U)control.Invoke(func, new object[] {control});
     }
     else
     {
         return func(control);
     }
}

然后让你写:

 t.setTitle(Global.GetControlPropertyThreadSafe(textBox1, t => t.Text));

您还可以使用相同的方法来获取和设置组合框项目。

于 2013-08-06T17:30:57.630 回答
2

您需要一个类似的 get 函数,如果需要,它会调用 Invoke。这取决于 GUI 的线程模型,该模型规定所有与 GUI 交互的方法只能由 GUI 线程调用。

如果您尝试阅读、写作或做任何其他事情,这并不重要。如果您不在 GUI 线程上,则必须使用 Invoke。

于 2013-08-06T17:33:21.267 回答