进度条工作但标签未更新

时间:2012-06-18 09:31:16

标签: c# winforms progress-bar uilabel

我有一个带接口的库。

Public Interface Progress
{
    int ProgressValue{get;set;},
    string ProgressText{get;set;},
}

库有一个方法Create(虚拟代码):

Public Class TestLibrary
{

    Progress _progress;

    Public void Create()
    {
        foreach(var n in TestList)
        {
            // Do Something
            _progress.ProgressValue = GetIndex(n);
            _progress.ProgressText = "Updating..." + n;
        }
    }
}

我有一个引用此库并调用Create方法的项目。它甚至实现了接口进度。

Public Class TestProject : Progress
{
    public int ProgressValue
    {
        get{return progressBar1.Value;}
        set{progressBar1.Value = value;}
    }

    public int ProgressText
    {
        get{return label1.Text;}
        set{label1.Text = value;}
    }
}

现在,当我运行应用程序时,进度条表现正常并正确显示进度,但label1的文本根本没有变化。但它在for循环结束时会发生变化并显示循环中的最后一项。任何人都可以帮助我吗?

注意:所有这些代码都是直接编写的,没有经过测试,因为我现在没有我的应用程序。对于任何语法错误,我们深表歉意。

3 个答案:

答案 0 :(得分:8)

听起来你的所有工作都是在UI线程上完成的。不要那样做。

相反,在后台线程中运行循环本身,并使用Control.InvokeControl.BeginInvoke(可能在Progress实现中)将调用封送到UI线程以更新用户界面。这将使您的UI在处理时保持响应(并且能够更新标签等)。

答案 1 :(得分:5)

使用Label instead of ProgressBar。您可以尝试使用此代码[using BackGroundWorker] -

using System.ComponentModel;
using System.Windows.Forms;

namespace WindowsFormsApplication1
{
    public partial class Form3 : Form
    {
        private BackgroundWorker _worker;
        BusinessClass _biz = new BusinessClass();
        public Form3()
        {
            InitializeComponent();
            InitWorker();
        }

        private void InitWorker()
        {
            if (_worker != null)
            {
                _worker.Dispose();
            }

            _worker = new BackgroundWorker
            {
                WorkerReportsProgress = true,
                WorkerSupportsCancellation = true
            };
            _worker.DoWork += DoWork;
            _worker.RunWorkerCompleted += RunWorkerCompleted;
            _worker.ProgressChanged += ProgressChanged;
            _worker.RunWorkerAsync();
        }


        void DoWork(object sender, DoWorkEventArgs e)
        {
            int highestPercentageReached = 0;
            if (_worker.CancellationPending)
            {
                e.Cancel = true;
            }
            else
            {
                double i = 0.0d;
                int junk = 0;
                for (i = 0; i <= 199990000; i++)
                {
                    int result = _biz.MyFunction(junk);
                    junk++;

                    // Report progress as a percentage of the total task.
                    var percentComplete = (int)(i / 199990000 * 100);
                    if (percentComplete > highestPercentageReached)
                    {
                        highestPercentageReached = percentComplete;
                        // note I can pass the business class result also and display the same in the LABEL  
                        _worker.ReportProgress(percentComplete, result);
                        _worker.CancelAsync();
                    }
                }

            }
        }

        void RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        {
            if (e.Cancelled)
            {
                // Display some message to the user that task has been
                // cancelled
            }
            else if (e.Error != null)
            {
                // Do something with the error
            }
        }

        void ProgressChanged(object sender, ProgressChangedEventArgs e)
        {
            label1.Text =  string.Format("Result {0}: Percent {1}",e.UserState, e.ProgressPercentage);
        }
    }

    public class BusinessClass
    {
        public int MyFunction(int input)
        {
            return input+10;
        }
    }
}

几天前发表同样的信息here

答案 2 :(得分:0)

您发布的代码使用一个线程。这意味着在上一个操作完成后,将依次执行每个操作。 由于您可以立即更新GUI元素,因此我假设代码将从主线程(也称为“ GUI线程”)运行。阻塞GUI线程会导致GUI(“图形用户界面”)在有空闲时间之前不会更新。

使用以下示例在每次迭代时刷新标签,以便更新您的UI。

label1.Text = i.ToString();
label1.Refresh();