我一直在寻找解决这个问题的两天以上,并最终决定提出这个问题。我找到了许多相关主题,但似乎没有一个能解决我的问题。最近,我尝试了所有列出的解决方案here。
背景信息:我有一个处理遍历大量数据的类。该类称为Traverse。有一个名为DoFullTraverse(Traverse.DoFullTraverse)的类方法,它运行一个完整的遍历而不是(取决于用户输入)最多需要30秒。我在WPF,MVVM模式工作。我想更新gui上的状态栏以了解DoFullTraverse的进度。我在函数的开头计算计算所需的确切循环数,然后递增循环计数器。每次达到另一个1/100时,我将进度条增加1.我的进度条(在xaml中)的值绑定到我的VM中名为PBarV
的属性。
最近的尝试:我尝试了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;
}
我认为我从根本上误解了后台工作者的工作方式。
主要问题:我可以让这个方法工作得很好,如果我把函数抛给后台工作者并像往常一样继续。问题是,在我的程序继续之前,我需要来自该函数的数据。因此,我需要它线性执行但仍然正确更新状态栏。
如果有人能够了解我所缺少的东西,或者甚至在正确的方向上推动我,我会非常感激。
编辑:这不重复。您提供的帖子不包括线性执行和等待后台工作人员在继续之前完成的问题。
编辑2 :(按照@Clemens请求)
我需要后台工作人员在主程序继续之前完成工作。我专门在后台工作程序中运行计算量很大的进程,以便可以更新进度条。但是,在主程序可以继续之前,我需要来自Traverse.DoFullTraverse();
非常具体。主程序应该暂停所有执行(除了更新状态栏),直到后台工作程序完成Traverse.DoFullTraverse();
答案 0 :(得分:1)
这是一个简单的例子,您可以使用并应用于您的视图模型。使用原型创建代码并了解其工作原理以便将其应用于更大,更复杂的应用程序非常重要。
请注意,该示例并不包含诸如如何实现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
绑定到按钮。