0

我正在增强通过 Web 服务与 SalesForce CRM 交互的 ac#win 表单应用程序。

我有以下代码可以对表单上的标签进行“线程安全”更新:

 delegate void SetTextCallback(string text);

 private void SetText(string text)
    {
        // InvokeRequired required compares the thread ID of the
        // calling thread to the thread ID of the creating thread.
        // If these threads are different, it returns true.
        if (this.lblAccessStatus.InvokeRequired)
        {
            SetTextCallback d = new SetTextCallback(SetText);
            this.Invoke(d, new object[] { text });
        }
        else
        {
            this.lblAccessStatus.Text = text;
            this.lblAccessStatus.Refresh();
        }
    }

我的表单(称为SFDetachifier.cs)有一个执行按钮,该按钮获取两个日历控件的日期,然后调用:

lblAccessStatus.Visible = true;
picProgress.Visible = true;

string fromDate = dtManualFromDate.Value.ToString("yyyy-MM-dd") + fromTime;
string toDate = dtManualToDate.Value.ToString("yyyy-MM-dd") + toTime;
string[] arg = new string[] { "S1", fromDate, toDate };
SFCtrl._SessionLog.Append("Manual detach requested on " + SFOrgName + "\n");
SFCtrl._SessionLog.Append("Date range requested: " + fromDate + " to " + toDate + "\n");

bgProcessing_Production.RunWorkerAsync(arg);

bgProcessing_Production具有后台工作人员的以下代码,其中包括对setText

private void bgProcessing_Production_DoWork(object sender, DoWorkEventArgs e)
    {
        String[] args = (String[])e.Argument;
        e.Result = args[0];

        // clear the datagrid view
        gvTaskCases.DataSource = null;
        //gvTaskCases.Rows.Clear();

        //if (gvTaskCases.Rows.Count != 0)
        //    {
        //    gvTaskCases.Rows.Clear(); // .Update();
        //    }

        SetText("login to SalesForce (on " + SFOrgName + ") ...please wait");

现在,当我运行我的应用程序时,我设置了一些日期,然后通过单击执行按钮执行上面的代码,它就可以正常工作(调用 SalesForce XML Web 服务)并将结果放入网格中。

一切正常,当我尝试再次运行该程序时出现问题(例如使用不同的日期)

然后我得到一个错误'跨线程操作无效:控制''从创建它的线程以外的线程访问。

我不明白,我有一个委托设置?

我做错了什么 - 我猜当后台工作人员再次运行时,它会在新线程上运行,因此线程不安全。

请问我该如何解决这个错误?

4

2 回答 2

1

您可以使用这样的匿名委托来缩短时间:

    private void removeGridDS()
    {
        this.gvTaskCases.Invoke((MethodInvoker)delegate
        {
            if (this.gvTaskCases.DataSource != null)
            {
                this.gvTaskCases.DataSource = null;
            }
        });
    }

    private void clear_gvTaskCases()
    {
        this.gvTaskCases.Invoke((MethodInvoker)delegate
        {
            if (this.gvTaskCases.Rows.Count != 0)
            {
                this.gvTaskCases.Rows.Clear();
            }
        });
    }

现在您不需要硬编码的委托。此外,您没有使用传入的布尔值。

于 2013-11-11T17:02:12.647 回答
0

这是我最终为了解决这个问题而做的事情,正如上面的评论之一所建议的那样,看起来很可能是对控件的调用gvTaskCases引发了错误,所以我也在该控件上创建了线程安全的方法调用:

delegate void SetTextCallback(string text);
delegate void clear_gvTaskCasesCallback(bool clearIt);
delegate void remove_gvTaskCasesDSCallback( bool removeDS);

private void removeGridDS(bool removeDS)
    {
        // InvokeRequired required compares the thread ID of the
        // calling thread to the thread ID of the creating thread.
        // If these threads are different, it returns true.
        if (this.gvTaskCases.InvokeRequired)
        {
        remove_gvTaskCasesDSCallback d = new remove_gvTaskCasesDSCallback(removeGridDS);
        this.Invoke(d, new object[] { removeDS });
        }
        else
        {
            if (this.gvTaskCases.DataSource !=null)
            {
            this.gvTaskCases.DataSource=null;
            }
        }
    }

    private void clear_gvTaskCases(bool clearIt)
    {
    // InvokeRequired required compares the thread ID of the
    // calling thread to the thread ID of the creating thread.
    // If these threads are different, it returns true.
    if (this.gvTaskCases.InvokeRequired)
    {
        clear_gvTaskCasesCallback d = new clear_gvTaskCasesCallback(clear_gvTaskCases);
        this.Invoke(d, new object[] { clearIt });
        }
        else
        {
        if (this.gvTaskCases.Rows.Count != 0)
        {
        this.gvTaskCases.Rows.Clear();
        }
    }
}
于 2013-11-11T15:13:53.180 回答