后台更新进度条的具体用例
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
绑定到一个按钮。
我找了两天多来解决这个问题,最后决定问这个问题。我找到了许多相关主题,但其中 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
绑定到一个按钮。