[英]Can a long-running method and a “Working…” dialog be run together using the Task Parallel Library to allow long task to write to a BindingList?
我有一个 WPF(C# 和 .NET 4)应用程序,其中有一个长时间运行的任务,它阻止了 UI,给人一种它已经挂起的印象。 我决定使用BackgroundWorker
线程将这个长时间运行的任务放到一个单独的线程中,并在单独的弹出窗口 window(下面命名为 WorkingDialog)中显示一个BusyIndicator
cator。 这工作正常,直到长时间运行的任务写入 BindingList(绑定到 UI 上的网格),我得到以下异常:
这种类型的 CollectionView 不支持从与 Dispatcher 线程不同的线程对其 SourceCollection 的更改
这是业务层代码(一个非常精简的版本)......
public class CustomMessage
{
public DateTime TimeStamp { get; private set; }
public string Message { get; private set; }
public CustomMessage(string message)
{
Message = message;
TimeStamp = DateTime.Now;
}
}
public class MyRandomBusinessClass
{
public BindingList<CustomMessage> LoggingList { get; private set; }
public MyRandomBusinessClass()
{
LoggingList = new BindingList<CustomMessage>();
}
public void SomeLongRunningTask()
{
System.Threading.Thread.Sleep(5000);
LoggingList.Add(new CustomMessage("Completed long task!"));
}
}
...用户界面...
<Window x:Class="WpfApplication4.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="MainWindow"
Height="350"
Width="525">
<StackPanel>
<DataGrid x:Name="logGrid"
AutoGenerateColumns="True"
ItemsSource="{Binding Path=LoggingList}" />
<Button Click="ProcessLongTask_Click"
Content="Do Long Task" />
</StackPanel>
</Window>
...用户界面代码...
using System.ComponentModel;
using System.Windows;
namespace WpfApplication4
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
public MyRandomBusinessClass RandomBusinessClass { get; set; }
public MainWindow()
{
InitializeComponent();
RandomBusinessClass = new MyRandomBusinessClass();
logGrid.DataContext = RandomBusinessClass;
}
private void ProcessLongTask_Click(object sender, RoutedEventArgs e)
{
ProcessTask1();
}
private void ProcessTask1()
{
WorkingDialog workingDialog = new WorkingDialog();
workingDialog.Owner = this;
BackgroundWorker worker = new BackgroundWorker();
worker.DoWork += delegate(object s, DoWorkEventArgs args)
{
RandomBusinessClass.SomeLongRunningTask();
};
worker.RunWorkerCompleted += delegate(object s, RunWorkerCompletedEventArgs args)
{
workingDialog.Close();
};
worker.RunWorkerAsync();
workingDialog.ShowDialog();
}
}
}
...工人对话...
<Window xmlns:extToolkit="http://schemas.microsoft.com/winfx/2006/xaml/presentation/toolkit/extended"
x:Class="WpfApplication4.WorkingDialog"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
mc:Ignorable="d"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Title="Working Dialog"
WindowStartupLocation="CenterOwner"
Height="116"
Width="199">
<Grid>
<extToolkit:BusyIndicator x:Name="busyIndicator"
IsBusy="True"
BusyContent="Working ..."
HorizontalAlignment="Left"
VerticalAlignment="Top" />
</Grid>
</Window>
我在这个论坛中发现了几个线程指向Bea Stollnitz的工作),但我想知道是否有一种方法可以执行长时间运行的任务,显示 WorkingDialog,然后在任务完成后使用任务并行库? 任务并行库能否在不修改业务层代码的情况下达到预期的效果?
编辑 - 解决方法 1
感谢 Ben 下面的评论,如果您的长时间运行的任务与 UI 在同一个程序集中或使用 System.Windows 命名空间,则有一种解决方法。 使用当前的应用程序调度程序,我可以毫无例外地将消息添加到我的列表中:
public class MyRandomBusinessClass
{
public BindingList<CustomMessage> LoggingList { get; private set; }
public MyRandomBusinessClass()
{
LoggingList = new BindingList<CustomMessage>();
}
public void SomeLongRunningTask()
{
System.Threading.Thread.Sleep(5000);
Dispatcher myDispatcher = Application.Current.Dispatcher;
myDispatcher.BeginInvoke((Action)(() => LoggingList.Add(new CustomMessage("Completed long task!"))));
}
}
我仍然很好奇这是否可以使用任务并行库或者我吠叫错误的树? 在我的实际应用程序中,长期运行的 class 位于单独的 class 库中。
编辑 2
我在下面接受了 Ben 的解决方案,因为它解决了我最初的问题,即在不阻塞 UI 或引发异常的情况下运行长时间运行的任务。 我很欣赏最初的问题是关于任务并行库的,所以如果您正在阅读这篇文章并认为它具有 TPL 解决方案,请原谅我。
您需要使用 调度程序将新元素添加到日志列表中。
public void SomeLongRunningTask()
{
System.Threading.Thread.Sleep(5000);
Application.Current.Dispatcher.BeginInvoke((Action)(() => LoggingList.Add(new CustomMessage("Completed long task!"))));
}
编辑:
public class MyRandomBusinessClass
{
public BindingList<CustomMessage> LoggingList { get; private set; }
public Action<MyRandomBusinessClass, CustomMessage> CallBackAction { get; private set; }
public MyRandomBusinessClass(Action<MyRandomBusinessClass, CustomMessage> cba)
{
LoggingList = new BindingList<CustomMessage>();
this.CallBackAction = cba;
}
public void SomeLongRunningTask()
{
System.Threading.Thread.Sleep(5000);
if (cba != null)
CallBackAction(this, new CustomMessage("Completed long task!"));
}
}
然后在哪里创建您的业务 class:
MyRandomBusinessClass businessClass = new MyRandomBusinessClass((sender, msg) =>
Application.Current.Dispatcher.BeginInvoke((Action)(() =>
sender.LoggingList.Add(msg))));
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.