ProgressBar从另一个线程更新

问题描述 投票:3回答:3

我知道这个问题之前曾被问过100次,但我读到的所有答案对我都没有用。所以,我会试试运气再问一遍。

我有一个SliderBarValueChanged事件上调用一个方法。在这个方法中,我做了一些需要时间的东西,我希望在这个时候用户会看到一个“工作”的ProgressBarIsIndeterminate=true)。

不幸的是,我没有成功使ProgressBar开始工作(在UI中),直到所有方法循环完成。我试过线程,BackgroundWorkerasync任务但没有成功..

我究竟做错了什么?

这是方法:

    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
    }
c# wpf multithreading progress-bar async-await
3个回答
4
投票

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 = ...                
    }));

    ....
}

1
投票

使用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);
  ...
}

请注意,您(几乎)不应在现代应用程序中使用DispatcherThreadBackgroundWorker


0
投票

我请你在StackOverflow上看看我对Progress Bar update from Background worker stalling问题的答案,而不是输入整个冗长的过程。它演示了如何正确使用BackgroundWorker对象来更新ProgressBar的进度值。

您可能还希望在MSDN上查看BackgroundWorker Class页面以获取更多详细信息。

© www.soinside.com 2019 - 2024. All rights reserved.