【问题标题】:How do I display progress during a busy loop?如何在繁忙的循环中显示进度?
【发布时间】:2010-11-14 17:49:30
【问题描述】:

我有一个从外部源读取大量数据的循环。该过程大约需要 20 秒,我想向用户显示进度。我不需要任何花哨的进度条,所以我选择将我的进度绘制在一个标签上,上面写着“Step 1/1000”,然后更改为“Step 2/1000”等。

我的代码如下所示:

// "count" is the number of steps in the loop, 
// I receive it in previous code

String countLabel = "/"+count.ToString();

for (i = 0; i < count; i++)
{
    ... do analysis ...
    labelProgress.Content = "Step "+i.ToString()+countLabel
}

但是,在该分析过程中,屏幕“卡住”并且进度未显示为前进。我从过去在 C++ 中理解这种行为,我可能会有一个单独的线程显示进度条,接收来自循环的通知,或某种形式的重绘/刷新,或强制窗口/应用程序处理其消息队列。

在 C# 中正确的做法是什么?我没有绑定到标签,所以如果有一个简单的进度条弹出屏幕,我可以用它来代替这个标签,那也很棒......

谢谢

【问题讨论】:

标签: c# wpf progress-bar


【解决方案1】:

UI 没有更新,因为您当前的线程具有比最终设置标签的 UI 线程更高的优先级;)。因此,在您的线程完成您的内容之前,它最终会更新您的标签。

幸运的是,每个 WPF 控件都有一个 Dispatcher 属性,可让您以另一个优先级启动新线程。

labelProgress.Dispatcher.Invoke(DispatcherPriority.Background,
                    () => labelProgress.Content = string.Format("Step {0}{1}", i, countLabel));

这会在后台启动一个线程并完成工作!您也可以尝试其他DispatcherPriority 选项

PS 我还冒昧地添加了一个匿名方法并在一定程度上修复了您的字符串解析..希望您不介意..

【讨论】:

    【解决方案2】:
        //Create a Delegate to update your status button
        delegate void StringParameterDelegate(string value);
        String countLabel = "/" + count.ToString();
        //When your button is clicked to process the loops, start a thread for process the loops
        public void StartProcessingButtonClick(object sender, EventArgs e)
        {
            Thread queryRunningThread = new Thread(new ThreadStart(ProcessLoop));
            queryRunningThread.Name = "ProcessLoop";
            queryRunningThread.IsBackground = true;
            queryRunningThread.Start();
        }
    
        private void ProcessLoop()
        {
            for (i = 0; i < count; i++)
            {
                ... do analysis ...
                UpdateProgressLabel("Step "+i.ToString()+countLabel);
            }
        }
    
        void UpdateProgressLabel(string value)
        {
            if (InvokeRequired)
            {
                // We're not in the UI thread, so we need to call BeginInvoke
                BeginInvoke(new StringParameterDelegate(UpdateProgressLabel), new object[] { value });
                return;
            }
            // Must be on the UI thread if we've got this far
            labelProgress.Content = value;
        }
    

    【讨论】:

    • 提供了BackgroundWorker,因此您不必处理所有从线程生成的混乱。
    • @sixlettervariables 这个方法的价值在于它允许你比仅仅将一个整数传回给调用者更灵活——你可以做任何事情(比如传回状态消息)跨度>
    【解决方案3】:

    将工作移至BackgroundWorker 并使用ReportProgress 方法。

    for (i = 0; i < count; i++)
    {
        ... do analysis ...
        worker.ReportProgress((100 * i) / count);
    }
    
    private void MyWorker_ProgressChanged(object sender,
        ProgressChangedEventArgs e)
    {
        taskProgressBar.Value = Math.Min(e.ProgressPercentage, 100);
    }
    

    【讨论】:

    • 是的,应该是。您的后台工作人员仅在一个线程中,进度条更新仅在 UI 线程中。如果 UI 线程以外的任何线程尝试更新值,它都会抛出异常。因此,使用此方法可以保证线程安全。
    猜你喜欢
    • 2016-07-29
    • 2017-01-18
    • 2016-04-02
    • 1970-01-01
    • 2021-12-20
    • 2015-07-31
    • 1970-01-01
    • 2012-03-28
    • 1970-01-01
    相关资源
    最近更新 更多