后台更新进度条的具体用例

Specific Use Case For Background Updating Progress Bar

我找了两天多来解决这个问题,最后决定问这个问题。我找到了许多相关主题,但其中 none 似乎解决了我的问题。最近,我尝试了 here.

列出的所有解决方案

背景信息:我有一个 class 可以处理大量数据。 class 称为遍历。有一个名为 DoFullTraverse (Traverse.DoFullTraverse) 的 class 方法,它运行一个完整的遍历最多可能需要 30 秒(取决于用户输入)。我在 WPF、MVVM 模式中工作。我想更新 gui 上的状态栏以了解 DoFullTraverse 的进度。我在函数的开头计算计算所需的确切循环次数,然后递增一个循环计数器。每次达到另一个 1/100 时,我将进度条递增 1。我的进度条(在 xaml 中)的值绑定到我名为 PBarV 的 VM 中的 属性。

最近的尝试: 我已经尝试了 100 种不同的解决方案,但我最近的尝试是这样的:

private void runTraverseAndUpdateBar()
{
    var worker = new BackgroundWorker();
    worker.DoWork += new DoWorkEventHandler(worker_DoWork);
    worker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(worker_Complete);
    worker.RunWorkerAsync();
    while (!ThreadCheck)
    {
        Thread.Sleep(500);
    }
}
void worker_DoWork(object sender, DoWorkEventArgs e)
{
    var worker = sender as BackgroundWorker;
    for (int i = 0; i < 36; i++)
    {
        Thread.Sleep(500);
        PBarV += 3;
    }
    e.Result = true;
}
void worker_Complete(object sender, RunWorkerCompletedEventArgs e)
{
    ThreadCheck = true;
}

我相信我从根本上误解了后台工作者的工作方式。

主要问题: 如果我将函数放入后台工作程序并照常继续,我可以使此方法正常工作。问题是,在我的程序继续之前,我需要来自该函数的数据。因此,我需要它线性执行但仍能正确更新状态栏。

如果有人能阐明我所缺少的东西,甚至能在正确的方向上推动我,我将不胜感激。

编辑:这不是重复的。您提供的 post 没有涵盖线性执行和等待后台工作程序完成后再继续的问题。

编辑 2:(根据@Clemens 的要求)

我需要后台工作者在主程序继续之前完成工作。我是 运行 后台工作者中的计算量大的进程,专门用于更新进度条。但是,在主程序可以继续之前,我需要来自 Traverse.DoFullTraverse();

的信息

非常具体。主程序应停止所有执行(除了更新状态栏)直到后台工作程序完成Traverse.DoFullTraverse();

这是一个简单的示例,您可以试用并应用于您的视图模型。使用原型来创建代码并了解其工作原理以便将其应用于更大、更复杂的应用程序非常重要。

请注意,该示例不包括诸如如何实现 INotifyPropertyChanged 和 ICommand 之类的琐碎内容——这些很容易做到。

另外,请注意 TraverseYo 中的注释。具体来说,就是告诉您当前所在线程的那些。了解跨线程的执行流程对于使其正常工作很重要。如果您不知道自己在哪个线程上,只需获取当前线程的 ApartmentState 即可。如果是 STA,您很可能在 UI 线程中。

public class LongLastingWorkViewModel : INotifyPropertyChanged
{
    public bool Busy 
    {
        // INotifyPropertyChanged property implementation omitted
    }
    public double PercentComplete 
    {
        // INotifyPropertyChanged property implementation omitted
    }

    public ICommand PerformWork { get; set; }

    public LongLastingWorkViewModel()
    {
        // delegated ICommand implementation omitted--there's TONS of it out there
        PerformWork = new DelegatedCommand(TraverseYo);
    }

    private void TraverseYo()
    {
        // we are on the UI thread here
        Busy = true;
        PercentComplete = 0;
        Task.Run(() => {
            // we are on a background thread here
            // this is an example of long lasting work 
            for(int i = 0; i < 10; i++)
            {
                Thread.Sleep(10 * 1000); // each step takes 10 seconds

                // even though we are on a background thread, bindings
                // automatically marshal property updates to the UI thread
                // this is NOT TRUE for INotifyCollectionChanged updates!
                PercentDone += .1; 
            }
            Busy = false;            
        });
}

您可以将 Busy 绑定到一个在执行运行时阻止所有 UI 的覆盖层,将 PercentComplete 绑定到一个进度条,将 PerformWork 绑定到一个按钮。