gpt4 book ai didi

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

转载 作者:太空宇宙 更新时间:2023-11-03 12:41:47 25 4
gpt4 key购买 nike

我一直在寻找这个问题的解决方案两天多了,最后决定问这个问题。我找到了许多相关主题,但似乎没有一个能解决我的问题。最近,我尝试了列出的所有解决方案 here .

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

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

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

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

编辑:这不是重复的。您提供的帖子未涵盖线性执行和等待后台工作人员完成后再继续的问题。

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

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

的信息

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

最佳答案

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

请注意,该示例不包括诸如如何实现 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 绑定(bind)到一个在执行运行时阻塞所有 UI 的覆盖层,将 PercentComplete 绑定(bind)到一个进度条,并将 PerformWork 绑定(bind)到一个按钮。

关于c# - 后台更新进度条的具体用例,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38742605/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com