2

我知道有很多类似的问题,我已经阅读了很多。不幸的是,在阅读完它们之后我仍然无法解决我的问题 - 但是我对 C# 比较陌生。根据文档,不是线程安全的问题导致InvalidOperationException使用调试器时出现问题。我的问题不是这种情况。

我用一个简单的原始测试类重新创建了这个问题,以专注于我的问题。

主窗体应该显示一种进度对话框。

public partial class ImportStatusDialog : Form
{
    public ImportStatusDialog()
    {
        InitializeComponent();
    }

    public void updateFileStatus(string path)
    {
        t_filename.Text = path;         
    }

    public void updatePrintStatus()
    {      
        t_printed.Text = "sent to printer";
    }

    public void updateImportStatus(string clientName)
    {
        t_client.Text = clientName;
    }

    public void updateArchiveStatus()
    {
        t_archived.Text = "archived";
    }
}

当从主窗体调用该代码而没有任何 Invoke() 时:

private void button1_Click(object sender, EventArgs e)
    {
        ImportStatusDialog nDialog = new ImportStatusDialog();

        nDialog.Show();

        nDialog.updateFileStatus("test");
        Thread.Sleep(1000);
        nDialog.updateImportStatus("TestClient");
        Thread.Sleep(1000);
        nDialog.updatePrintStatus();
        Thread.Sleep(1000);
        nDialog.updateArchiveStatus();
        Thread.Sleep(1000);

        nDialog.Close();
    }

即使我这样称呼它:

private void button3_Click(object sender, EventArgs e)
    {
        ImportStatusDialog nDialog = new ImportStatusDialog();

        nDialog.Show();

        if (this.InvokeRequired)
        {
            this.Invoke((MethodInvoker)delegate
            {
                nDialog.updateFileStatus("Test");
            });
        }
        else
        {
            nDialog.updateFileStatus("Test");
        }

        Thread.Sleep(1000);

        if (this.InvokeRequired)
        {
            this.Invoke((MethodInvoker)delegate
            {
                nDialog.updatePrintStatus();
            });
        }
        else
        {
            nDialog.updatePrintStatus();
        }

        Thread.Sleep(1000);

        if (this.InvokeRequired)
        {
            this.Invoke((MethodInvoker)delegate
            {
                nDialog.updateImportStatus("cName");
            });
        }
        else
        {
            nDialog.updateImportStatus("cName");
        }

        Thread.Sleep(1000);

        if (this.InvokeRequired)
        {
            this.Invoke((MethodInvoker)delegate
            {
                nDialog.updateArchiveStatus();
            });
        }
        else
        {
            nDialog.updateArchiveStatus();
        }

        Thread.Sleep(1000);

        nDialog.Close();
    }

在设计器中看起来像这样的对话框(在我的示例中)

设计师观点

会这样显示:

运行时对话框

当我使用ShowDialog()而不是Show()时,对话框会正确显示,但正如 API Doc 指出的那样

您可以使用此方法在应用程序中显示模式对话框。调用此方法时,它后面的代码直到对话框关闭后才会执行

这不是我想要的,特别是因为这意味着对话框更新只会在它再次关闭后才会发生。

我在这里做错了什么?这似乎是一个微不足道的问题,但解决方案却避开了我。请记住,我是 C# GUI 编程的新手。

另外,我想问一下使用 Invoke() 的正确位置是什么?你会在调用对话框方法时在主窗体中使用它还是在对话框更新方法本身中使用它?

4

1 回答 1

2

看起来您在这里没有使用多个线程,因此不需要调用内容。只有跨线程调用才需要调用 - 您正在创建多个表单,但所有代码都在同一个线程中运行。

如果您在主 UI 线程中进行工作(正如按钮单击事件中的代码所暗示的那样),那么只需Application.DoEvents()定期调用以允许刷新进度表单。

更好的解决方案是将 aBackgroundWorker用于您的工作,并让它定期报告其进度。

然后你可以在BackgroundWorker的ProgressChanged事件中更新进度表(它会在主线程中执行,所以你仍然不需要调用任何东西)。

于 2012-08-28T22:11:38.557 回答