在任务运行时以另一种形式更新进度条

时间:2015-02-13 20:39:23

标签: c# winforms c#-5.0

**最终我将同时运行四个任务,并有另一个包含四个进度条的表单。我希望每个进度条在其工作任务完成时进行更新。

这是我试图为初学者做的事情。

我有一个表格上有一些按钮。当我点击一个时,我创建一个新任务来做一些工作。

public partial class MyMainForm : Form
{

    private void btn_doWork_Click(object sender, EventArgs e)
    {
        Task task = new Task(RunComparisons);
        task.Start();
    }

    private void RunComparisons()
    {
        int progressBarValue = 0;
        MyProgressBarForm pBar = new MyProgressBarForm(maxValue, "some text");
        pBar.ShowDialog();
        foreach(string s in nodeCollection)
        {
            //do some work here
            progressBarValue++;
            pBar.updateProgressBar(progressBarValue, "some new text");
        }
        pBar.BeginInvoke(new Action(() => pBar.Close()));
    }
}

在另一个包含带进度条的表单的类中:

public partial class MyProgressBarForm : Form
{
    public MyProgressBarForm(int maxValue, string textToDisplay)
    {
        InitializeComponent();
        MyProgressBarControl.Maximum = maxValue;
        myLabel.Text = textToDisplay;
    }

    public void updateProgressBar(int progress, string updatedTextToDisplay)
    {
        MyProgressBarForm.BeginInvoke(
            new Action(() =>
            {
                MyProgressBarControl.Value = progress;
                myLabel.Text = updatedTextToDisplay;
            }));
    }

单击“执行”按钮时,将显示进度条表单,但不会更新。它只是坐在那里挂起。如果我注释掉pBar.ShowDialog();然后进度条表格不显示,但要完成的工作完全完成。

当我创建自己的线程时,我的工作非常完美,但是我读到了有关任务的内容,现在我试图让它以这种方式运行。我哪里出错了?

3 个答案:

答案 0 :(得分:15)

TPL添加了IProgress接口,用于根据长时间运行的非UI操作的进度更新UI。

您需要做的就是在UI中创建一个Progress实例,其中包含如何使用进度更新它的说明,然后将其传递给您的工作人员,该工作人员可以通过它报告进度。

public partial class MyMainForm : System.Windows.Forms.Form
{
    private async void btn_doWork_Click(object sender, EventArgs e)
    {
        MyProgressBarForm progressForm = new MyProgressBarForm();
        progressForm.Show();
        Progress<string> progress = new Progress<string>();
        progress.ProgressChanged += (_, text) =>
            progressForm.updateProgressBar(text);
        await Task.Run(() => RunComparisons(progress));
        progressForm.Close();
    }
    private void RunComparisons(IProgress<string> progress)
    {
        foreach (var s in nodeCollection)
        {
            Process(s);
            progress.Report("hello world");
        }
    }
}
public partial class MyProgressBarForm : System.Windows.Forms.Form
{
    public void updateProgressBar(string updatedTextToDisplay)
    {
        MyProgressBarControl.Value++;
        myLabel.Text = updatedTextToDisplay;
    }
}

这使得Progress Form处理显示UI的进度,工作代码只处理执行工作,主要表单只需创建进度表单,启动工作,并在完成后关闭表单,它将全部保留通过UI线程跟踪进度和marhsaling到Progress的工作。它还避免了多个UI线程;您当前从非UI线程创建和操作UI组件的方法会产生许多问题,这些问题会使代码变得复杂并使其难以维护。

答案 1 :(得分:0)

在父表单的主UI线程上创建进度条表单,然后在按钮单击事件中调用对象上的Show()方法。

这是一个有2个小节的例子:

//In parent form ...
private MyProgressBarForm progressBarForm = new MyProgressBarForm();

private void button1_Click(object sender, EventArgs e)
{
    progressBarForm.Show();
    Task task = new Task(RunComparisons);
    task.Start();
}

private void RunComparisons()
{
    for (int i = 1; i < 100; i++)
    {
        System.Threading.Thread.Sleep(50);
        progressBarForm.UpdateProgressBar(1, i);
    }
}

//In MyProgressBarForm ...
public void UpdateProgressBar(int index, int value)
{
    this.Invoke((MethodInvoker) delegate{
        if (index == 1)
        {
            progressBar1.Value = value;
        }
        else
        {
            progressBar2.Value = value;
        }
    });
}

答案 2 :(得分:-1)

.ShowDialog是一个阻止通话;执行不会继续,直到对话框返回结果。您应该查看BackgroundWorker来处理另一个线程上的工作并更新对话框。