ProgressBar从另一个线程更新
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了ProgressBar从另一个线程更新相关的知识,希望对你有一定的参考价值。
我知道这个问题之前曾被问过100次,但我读到的所有答案对我都没有用。所以,我会试试运气再问一遍。
我有一个SliderBar
在ValueChanged
事件上调用一个方法。在这个方法中,我做了一些需要时间的东西,我希望在这个时候用户会看到一个“工作”的ProgressBar
(IsIndeterminate=true
)。
不幸的是,我没有成功使ProgressBar
开始工作(在UI中),直到所有方法循环完成。我试过线程,BackgroundWorker
和async
任务但没有成功..
我究竟做错了什么?
这是方法:
private void Slider_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e){
WorkingBar.IsIndeterminate = true; //change the progressBar
SqlAThread = new Thread(SqlAStart);
SqlAThread.Start();
}
线程:
private void SqlAStart()
{
... //do some stuff
}
在Slider_ValueChanged
你开始了一个新的步骤来运行耗时的SqlAStart()
方法,但SqlAStart
立即将工作推回到UI线程(通过Dispatcher.Invoke()
)。因此,您的UI会挂起,并且在完成工作之前您不会看到任何ProgressBar
进度。
在SqlAStart
,只需要更新进度条的Dispatcher.Invoke()
:
private void SqlAStart()
{
ServerThread = new Thread(HttpSrv.listen);
ServerThread.Start();
...
this.Dispatcher.Invoke((Action)(() => {
WorkingBar.Value = ...
}));
....
}
使用IProgress<T>
这很容易:
private async void Slider_ValueChanged(object sender, RoutedPropertyChangedEventArgs<double> e)
{
if (State.Value > 0)
{
var progress = new Progress<int?>(value =>
{
WorkingBar.IsIndeterminate = (value == null);
if (value != null)
WorkingBar.Value = value.Value;
});
await Task.Run(() => doWork(progress));
Task.Run(SqlAStart);
}
...
}
private void SqlAStart(IProgress<int?> progress)
{
ServerTask = Task.Run(HttpSrv.listen);
...
}
private void doWork(IProgress<int?> progress)
{
if (progress != null)
progress.Report(null);
...
}
请注意,您(几乎)不应在现代应用程序中使用Dispatcher
,Thread
或BackgroundWorker
。
我请你在StackOverflow上看看我对Progress Bar update from Background worker stalling问题的答案,而不是输入整个冗长的过程。它演示了如何正确使用BackgroundWorker
对象来更新ProgressBar
的进度值。
您可能还希望在MSDN上查看BackgroundWorker
Class页面以获取更多详细信息。
以上是关于ProgressBar从另一个线程更新的主要内容,如果未能解决你的问题,请参考以下文章