如何正确实现带有 ProgressBar 更新的 BackgroundWorker?

Posted

技术标签:

【中文标题】如何正确实现带有 ProgressBar 更新的 BackgroundWorker?【英文标题】:How to correctly implement a BackgroundWorker with ProgressBar updates? 【发布时间】:2013-10-20 12:06:21 【问题描述】:

-更新--14/10 也问过这个question

为了清楚地了解正在发生的事情并考虑到 cmets 和这篇文章here

我现在真正想做的是调用一个带有进度条的新表单,并在我的后台线程将我的长进程运行到数据库时运行它并制作动画,然后调用关闭表单事件

后台worker设置在这里

 public partial class MainWindow : Window

    //Declare background workers
    BackgroundWorker bw = new BackgroundWorker();
    BackgroundWorker bwLoadCSV = new BackgroundWorker();
    BackgroundWorker bwProgressBar = new BackgroundWorker();

然后在这里添加代表

  public MainWindow()
    
        bwLoadCSV.WorkerReportsProgress = true;
        bwLoadCSV.WorkerSupportsCancellation = true;
        bwLoadCSV.DoWork += new DoWorkEventHandler(bwLoadCSV_DoWork);
        bwLoadCSV.ProgressChanged += new ProgressChangedEventHandler(bwLoadCSV_ProgressChanged);
        bwLoadCSV.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bwLoadCSV_RunWorkerCompleted);

从主窗口类在这里调用

  private void CSV_Load_Click(object sender, RoutedEventArgs e)
    ///Function to read csv into datagrid
    ///
    
        //Turn Cursor to wait
        System.Windows.Forms.Cursor.Current = System.Windows.Forms.Cursors.WaitCursor;

        //Test connection to sql server
        if (CHHoursDataProvider.IsDatabaseOnline() == false)
        
            System.Windows.Forms.MessageBox.Show("Can not establish contact with sql server" + "\n" + "Contact IT", "Connection Error");
            //Set UI picture
            return;
        
        //Set a control to update the user here
        tbLoadDgStat.Visibility = Visibility.Visible;

        tbLoadDgStat.Text = "Getting data templete from Database...";
        string FilePath = txFilePath.Text;
        if (bwLoadCSV.IsBusy != true)
        
            //load the context object with parameters for Background worker
            bwCSVLoadContext Context = new bwCSVLoadContext();
            Context.Site = cBChSite.Text;
            Context.FilePath = txFilePath.Text;
            Context.FileName = fileTest;
            Context.Wageyear = cbWageYear.Text;
            Context.Startdate = ((DateTime)dpStartDate.SelectedDate);
            Context.Enddate = ((DateTime)dpEndDate.SelectedDate);

            bwLoadCSV.RunWorkerAsync(Context);                

        

后台工作者做的工作是

private void bwLoadCSV_DoWork(object sender, DoWorkEventArgs e)
    
        BackgroundWorker worker = sender as BackgroundWorker;

        bwCSVLoadContext Context = e.Argument as bwCSVLoadContext;


        worker.ReportProgress((10));
        if ((worker.CancellationPending == true))
        
            e.Cancel = true;

        
        else
        
            // Perform a time consuming operation and report progress load csv into datagrid.

为了报告后台工作,我这样做了。这是我尝试加载一个名为 ProgressDialog 的新表单的地方,它上面有一个进度条,我尝试将其设置为 Indeterminable,因此它只是在我的 ProgressDialoge 表单上“晃动”以向用户显示仍在进行的事情。我使用了后台工作的记者部分,因为我相信它可以访问主窗口线程,我希望随后从主窗口线程调用调用方法,但我不太确定

这是记者

 private void bwLoadCSV_ProgressChanged(object sender, ProgressChangedEventArgs e)

    


        this.Dispatcher.Invoke((MethodInvoker)delegate  tbLoadDgStat.Visibility = Visibility.Visible; );
        //tbLoadDgStat.Visibility = Visibility.Visible;


        //this.progressBar1.Value = e.ProgressPercentage;//This works but pauses on long steps
        if (e.ProgressPercentage == 10)
        
            //Try to open a new form with a class ProgressDialog and set the progressbar
            // on the frm to IsIndeterminate=true
            //THIS IS NOT WORKING
            this.Dispatcher.BeginInvoke (new Action(() =>
                ProgressDialog progressDialog = new ProgressDialog();
                progressDialog.SetIndeterminate(true);
            ));

             //this updates the main form OK
             this.Dispatcher.Invoke((MethodInvoker)delegate  tbLoadDgStat.Text = "Getting data templete from Database..."; );

        
        else if (e.ProgressPercentage == 20)
        

            this.Dispatcher.Invoke((MethodInvoker)delegate  tbLoadDgStat.Text = "Data template retrieved..."; );

        
        else
        

            if (e.ProgressPercentage % 10 == 0)
            
                this.Dispatcher.Invoke((MethodInvoker)delegate  tbLoadDgStat.Text = "Adding Data..." + e.ProgressPercentage.ToString() + "%"; );
            
        

最后是 ProgressDialog 表单的 xaml 和它的类

<Window x:Class="Test_Read_CSV.ProgressDialog"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="Progress Dialog" Height="115" Width="306" Name="ProgressPopup">
<Grid>
    <ProgressBar Height="31" HorizontalAlignment="Left" Margin="12,33,0,0" Name="progressBar1" VerticalAlignment="Top" Width="250" />
    <TextBox Height="23" HorizontalAlignment="Left" Margin="7,4,0,0" Name="tbEvent" VerticalAlignment="Top" Width="254" IsReadOnly="True" IsEnabled="False" />
</Grid>

/// <summary>
/// Interaction logic for ProgressDialog.xaml
/// </summary>
public partial class ProgressDialog : Window

    public ProgressDialog()
    
        WindowStartupLocation = WindowStartupLocation.CenterScreen;
        InitializeComponent();
    

    public ProgressDialog(String Purpose)
    
        InitializeComponent();
        tbEvent.Text = Purpose;
        WindowStartupLocation = WindowStartupLocation.CenterScreen;

    
    public void UpdateProgress(int progress)
    
        progressBar1.Dispatcher.BeginInvoke(
            new Action(() =>
            
                progressBar1.Value = progress;
            
        ));
    

    public void SetIndeterminate(bool isIndeterminate)
    
        progressBar1.Dispatcher.BeginInvoke(
            new Action(() =>
            
                if (isIndeterminate)
                
                    progressBar1.IsIndeterminate = true;
                
                else
                
                    progressBar1.IsIndeterminate = false;
                
            
        ));
    



我已经阅读并完成了一些关于后台工作者的教程,甚至一些关于线程的教程,但似乎无法得到我想要的结果

我的想法是我有两个漫长的过程,我要么从远程 bd 获取数据表克隆,要么从我的 wpf 应用程序 (.net 4) 更新数据库。在进程运行时,我想要一个进度条控件并对其进行更新,原因很明显,要明确一些工作正在进行中。所以我在后台工作人员中执行了通常的报告进度例程,它可以工作....但是,在我的 dowork 线程中,我有这个命令

CHHoursDataProvider CH = new CHHoursDataProvider();
oTable = CH.CloneCHHours();

这是与 db 的通信所在,并且此命令在 *** 远程连接上需要 60 - 90 秒,所以即使我这样做

CHHoursDataProvider CH = new CHHoursDataProvider();
worker.ReportProgress((10));
oTable = CH.CloneCHHours();
worker.ReportProgress((20));

主窗口看起来仍然冻结,就像崩溃了一样!

所以我要做的就是在开始调用后台工作时设置一个正在运行的进度条并让它一直运行到任务结束。这就是我完成我的第一个项目所需要做的全部事情,三天后我仍然无法理解它!

所以我尝试了跟随

在bw进度改变和主窗口类中

 this.progressBar2.IsIndeterminate = true;

但是动画直到 Dowork 线程完成后才开始。

然后我创建了另一个后台工作人员来更新进度条2,它链接到主窗口上的一个按钮是可以的,但是当我尝试从另一个后台工作人员或主窗口类使用它时没有运行直到 dowork 线程在第一个后台工作线程上完成

然后我尝试遵循调用方法,但真的迷路了!

所以任何人都可以帮助我猜想这与线程和处理错误的线程等有关,但我对此做了什么我不知道。

我可以根据需要发布更多代码

伊恩

【问题讨论】:

听起来代码仍在 UI 线程上运行,因此导致它冻结。从问题中无法猜测这是如何发生的,CHHoursDataProvider 是一个黑匣子。使用 Debug + Windows + Threads 调试器窗口进行诊断。 除非调用将后台线程的请求编组到主 gui 线程的调用,否则无法从后台线程更新主 gui 线程。 @HansPassant 该代码没有运行 UI 线程,而是该步骤 codeCHHoursDataProvider CH = new CHHoursDataProvider(); oTable = CH.CloneCHHours();code 到远程数据库获取数据表定义并返回。此步骤可能需要 90 秒。所以进度条的更新从 0 到 10% 然后等待 90 秒然后进度条更新更新 10% 到 20%。但是此时主窗口看起来像是崩溃了 【参考方案1】:

由于您没有展示完整的BackgroundWorker 代码,我无法判断您是否正确实现了它。因此,我所能做的就是向您展示一个更新ProgressBar 控件的简单工作示例:

UserControlXAML:

<UserControl x:Class="WpfApplication1.Views.TestView"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
    mc:Ignorable="d" 
    d:DesignHeight="300" d:DesignWidth="300" Loaded="UserControl_Loaded">
    <ProgressBar x:Name="progressBar" Height="25" Margin="20" Minimum="0" 
        Maximum="50" />
</UserControl>

MainWindowXAML:

<Window x:Class="WpfApplication1.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:Views="clr-namespace:WpfApplication1.Views"
    Title="MainWindow" Height="350" Width="525">
    <Views:TestView />
</Window>

UserControl后面的代码:

using System.ComponentModel;
using System.Threading;
using System.Windows;
using System.Windows.Controls;

namespace WpfApplication1.Views

    public partial class TestView : UserControl
    
        private BackgroundWorker backgroundWorker = new BackgroundWorker();

        public TestView()
        
            InitializeComponent();
            backgroundWorker.WorkerReportsProgress = true;
            backgroundWorker.ProgressChanged += ProgressChanged;
            backgroundWorker.DoWork += DoWork;
            // not required for this question, but is a helpful event to handle
            backgroundWorker.RunWorkerCompleted += BackgroundWorker_RunWorkerCompleted;
        

        private void UserControl_Loaded(object sender, RoutedEventArgs e)
        
            backgroundWorker.RunWorkerAsync();
        

        private void DoWork(object sender, DoWorkEventArgs e)
        
            for (int i = 0; i <= 100; i++)
            
                // Simulate long running work
                Thread.Sleep(100);
                backgroundWorker.ReportProgress(i);
            
        

        private void ProgressChanged(object sender, ProgressChangedEventArgs e)
        
            // This is called on the UI thread when ReportProgress method is called
            progressBar.Value = e.ProgressPercentage;
        

        private void BackgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        
            // This is called on the UI thread when the DoWork method completes
            // so it's a good place to hide busy indicators, or put clean up code
        
    

【讨论】:

谢谢谢里登。这更像是对进度条的更新正在工作的情况,而是步骤 codeCHHoursDataProvider CH = new CHHoursDataProvider(); oTable = CH.CloneCHHours();code 到远程数据库获取数据表定义并返回。此步骤可能需要 90 秒。所以进度条的更新从 0 到 10% 然后等待 90 秒然后进度条更新更新 10% 到 20%。但是此时主窗口看起来像是崩溃了 如果我想使用 ViewModel 类中的代码而不是类后面的代码,如何绑定加载的属性?它会像 绑定是一个单独的问题......但是,一旦您拥有数据绑定属性,只需在上面提到的 UI 线程上调用的两种方法中的任何一种中更新其值。参考您的 How to bind Progressbar and Datagrid at the same time using MVVM? 问题,您只需从这些方法更新您的 CurrentProgress 属性,所以不,您不会处理 ProgressBar 事件(或任何其他事件)查看模型。 [ProgressBar 控件 - 完整的 WPF 教程] (wpf-tutorial.com/misc-controls/the-progressbar-control) 这不是生产代码。这是一个简化的示例,该行应该模拟一个长时间运行的过程。

以上是关于如何正确实现带有 ProgressBar 更新的 BackgroundWorker?的主要内容,如果未能解决你的问题,请参考以下文章

Inno Setup - 正确定时更新进度条

Qt5 - QML:如何正确连接 ProgressBar 和 Button 以进行长时间运行的循环计算

DialogFragment 和 ProgressBar 在任务更新时更新

如何显示ProgressBar的进度条,给分100

下载文件时如何更新UICollectionViewCell子类中的progressBar

如何在QML中使用按钮,GroupBox,文本和ProgressBar设置基本的GridLayout