我有一个后台线程来处理与外部服务的通信。每次后台线程收到一条消息时,我都想将其传递给 UI 线程以进行进一步处理(显示给用户)。
目前我已经创建了一个线程安全的消息队列,它定期在 Timer.Tick 中汇集并填充到后台线程中。但是这个解决方案是次优的。
你知道如何使用消息泵将事件从后台线程传递到 ui 线程吗?
我有一个后台线程来处理与外部服务的通信。每次后台线程收到一条消息时,我都想将其传递给 UI 线程以进行进一步处理(显示给用户)。
目前我已经创建了一个线程安全的消息队列,它定期在 Timer.Tick 中汇集并填充到后台线程中。但是这个解决方案是次优的。
你知道如何使用消息泵将事件从后台线程传递到 ui 线程吗?
您可以使用 Control.Invoke 并使用委托。委托将在创建控件的线程上执行。
有一些技巧。
Control.Invoke()
(等人)
我发现这种 winforms 技术始终易于使用,但请注意,您需要一些微妙的规则才能正确使用。我试图捕捉一个通用的、有效的实现,它可以正确处理我在 stackoverflow 其他地方发布的代码段中的规则。
我不需要太多使用这种技术,所以我真的不能说任何有意义的东西。但是,您应该知道它存在。我相信这是确保在特定线程的上下文中调用某些内容的有效方法,即使该线程不是ui 线程。
如果您使用 WPF,则 WPF 控件通常派生自DispatcherObject
提供 Dispatcher 对象。这是一种功能更丰富的同步技术Control.Invoke()
,但也更复杂。请务必仔细阅读文档。
下面是在 WPF 中将 Dispacther 对象与 MSMQ 一起使用的示例。
后面的代码:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using System.Messaging;
namespace MSMQGui
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
private string queueName = @".\private$\MyFunWithMSMQ";
private MessageQueue queue = null;
public MainWindow()
{
InitializeComponent();
if (!MessageQueue.Exists(queueName))
MessageQueue.Create(queueName,false);
queue = new MessageQueue(queueName);
queue.ReceiveCompleted += receiveCompleted;
}
private void btnAddMessage_Click(object sender, RoutedEventArgs e)
{
string message = txtMessage.Text;
txtMessage.Text = String.Empty;
queue.Send(message);
MessageBox.Show("Message :" + message + " sent");
}
private void Populate(object sender, RoutedEventArgs e)
{
try
{
queue.BeginReceive(TimeSpan.FromSeconds(1)) ;
}
catch (MessageQueueException)
{
MessageBox.Show("No message available");
}
}
private void receiveCompleted(object source, ReceiveCompletedEventArgs e)
{
try
{
var message=queue.EndReceive(e.AsyncResult);
Action<string> addMessage= (string msg) => {
ListViewItem item = new ListViewItem();
item.Content = msg;
lsvMessages.Items.Add(item);
};
this.Dispatcher.Invoke(addMessage, message.Body as string);
}
catch (MessageQueueException)
{
MessageBox.Show("No message available");
}
}
}
}
XAML:
<Window x:Class="MSMQGui.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">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="1*"></ColumnDefinition>
<ColumnDefinition Width="3*"></ColumnDefinition>
</Grid.ColumnDefinitions>
<Grid.RowDefinitions>
<RowDefinition Height="1*"></RowDefinition>
<RowDefinition Height="9*"></RowDefinition>
<RowDefinition Height="1*"></RowDefinition>
</Grid.RowDefinitions>
<!-- First row -->
<Label x:Name="lblMessage"
Content="Message:"
HorizontalAlignment="Stretch"
VerticalAlignment="Top"
HorizontalContentAlignment="Right"
Grid.Column="0" Grid.Row="0"
></Label>
<StackPanel Grid.Column="1" Grid.Row="0" Orientation="Horizontal">
<TextBox x:Name="txtMessage" Width="200" HorizontalAlignment="Left" ></TextBox>
<Button x:Name="btnAddMessage" Content="Add message" Margin="5,0,0,0" Click="btnAddMessage_Click"></Button>
</StackPanel>
<!-- Second row -->
<ListView x:Name="lsvMessages" Grid.Column="0" Grid.ColumnSpan="2" Grid.Row="1" HorizontalAlignment="Stretch" VerticalAlignment="Stretch" Margin="0,5,0,0">
</ListView>
<!-- Third row-->
<Button x:Name="btnPopulate" Grid.Column="1" Grid.Row="2" HorizontalAlignment="Right" Click="Populate" Content="Get messages from queque" Margin="5,0,0,0"></Button>
</Grid>
</Window>
如果您的 GUI 线程已阻塞且未处理任何消息,您可以使用Application.DoEvents
强制 GUI 线程处理该线程上的所有等待消息。
要将消息泵送到 Control 的线程,当然可以使用Control.BeginInvoke
orControl.Invoke
方法,但请注意,Control.Invoke
如果 的拥有线程Control
当前正在阻塞,则会阻塞。
您也可以在 WindowsBase.dll 中使用 WPF Dispatcher(类 Dispatcher)。