Как правильно реализовать BackgroundWorker с обновлениями ProgressBar?
-Обновлено -14/10 также задали этот вопрос
Чтобы дать четкое представление о том, что происходит и с учетом комментариев и из этой статьи здесь
Что я действительно хочу сейчас сделать, так это вызвать новую форму с индикатором выполнения и запустить и анимировать ее, пока мой фоновый поток запускает мой длинный процесс в базе данных, а затем вызывать событие закрытия формы
Фоновый рабочий настроен здесь
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, чтобы показать пользователю, что что-то еще происходит. Я использовал репортерную часть фоновой работы, потому что я считаю, что он имеет доступ к потоку главного окна, и я надеюсь, что метод invoke затем вызывается из потока основного окна, но я не совсем уверен
Вот репортер
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() + "%"; });
}
}
Наконец, xaml для формы ProgressDialog и ее класса
<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;
}
}
));
}
}
}
Я прочитал и сделал несколько учебников по фоновому рабочему и даже по темам, но не могу получить желаемый результат
Идея в том, что у меня есть два длинных процесса, в которых я либо получаю клон данных с моего удаленного диска, либо обновляю базу данных из моего приложения wpf (.net 4). Пока процесс запущен, мне нужен элемент управления индикатора выполнения и его обновление, по очевидной причине, чтобы дать понять, что какая-то работа продолжается. Так что я сделал обычные процедуры выполнения отчетов в фоновом режиме, и он работает.... Однако в моей ветке dowork у меня есть эта команда
CHHoursDataProvider CH = new CHHoursDataProvider();
oTable = CH.CloneCHHours();
это где связь с БД и эта команда занимает хорошие 60 - 90 секунд на удаленном соединении vpn, так что даже если я делаю это
CHHoursDataProvider CH = new CHHoursDataProvider();
worker.ReportProgress((10));
oTable = CH.CloneCHHours();
worker.ReportProgress((20));
Главное окно все еще выглядит замороженным и, как будто оно разбилось!
Поэтому все, что я хочу сделать, - это в начале вызова фоновой работы установить индикатор выполнения и оставить его включенным до конца задачи. Это все, что мне нужно сделать, чтобы закончить свой первый в истории проект, и через три дня я все еще не могу разобраться с ним!
Итак, я попробовал следующее
В bw прогресс изменился и в главном окне класс
this.progressBar2.IsIndeterminate = true;
Однако анимация не начинается, пока не закончится поток Dowork.
Затем я создал еще один фоновый рабочий для обновления progressbar2, который связывался с кнопкой в главном окне, и все было в порядке, но как только я попытался использовать его из другого фонового рабочего или из основного окна, класс не запускался до окончания работы. поток завершен на первом фоне рабочего
Затем я попытался следовать методу вызова, но ДЕЙСТВИТЕЛЬНО заблудился на этом!
Так что кто-нибудь может помочь, я могу догадаться, что это что-то, связанное с многопоточностью и работой над неправильной нитью и т. Д., Но что я делаю с этим, я понятия не имею.
Я могу разместить больше кода по мере необходимости
Ян
1 ответ
Как вы еще не показали свой полный BackgroundWorker
код, я не могу сказать, правильно ли вы его реализовали. Таким образом, все, что я могу сделать, это показать вам простой рабочий пример обновления ProgressBar
управления:
UserControl
XAML:
<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>
MainWindow
XAML:
<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
}
}
}