我有一个应用程序,每当我正在观看的日志文件以下列方式更新时,我的数据网格都会更新(附加新文本):
private void DGAddRow(string name, FunctionType ft)
{
ASCIIEncoding ascii = new ASCIIEncoding();
CommDGDataSource ds = new CommDGDataSource();
int position = 0;
string[] data_split = ft.Data.Split(' ');
foreach (AttributeType at in ft.Types)
{
if (at.IsAddress)
{
ds.Source = HexString2Ascii(data_split[position]);
ds.Destination = HexString2Ascii(data_split[position+1]);
break;
}
else
{
position += at.Size;
}
}
ds.Protocol = name;
ds.Number = rowCount;
ds.Data = ft.Data;
ds.Time = ft.Time;
dataGridRows.Add(ds);
rowCount++;
}
...
private void FileSystemWatcher()
{
FileSystemWatcher watcher = new FileSystemWatcher(Environment.CurrentDirectory);
watcher.Filter = syslogPath;
watcher.NotifyFilter = NotifyFilters.LastAccess | NotifyFilters.LastWrite
| NotifyFilters.FileName | NotifyFilters.DirectoryName;
watcher.Changed += new FileSystemEventHandler(watcher_Changed);
watcher.EnableRaisingEvents = true;
}
private void watcher_Changed(object sender, FileSystemEventArgs e)
{
if (File.Exists(syslogPath))
{
string line = GetLine(syslogPath,currentLine);
foreach (CommRuleParser crp in crpList)
{
FunctionType ft = new FunctionType();
if (crp.ParseLine(line, out ft))
{
DGAddRow(crp.Protocol, ft);
}
}
currentLine++;
}
else
MessageBox.Show(UIConstant.COMM_SYSLOG_NON_EXIST_WARNING);
}
当为FileWatcher引发事件时,因为它创建了一个单独的线程,当我尝试运行dataGridRows.Add(ds)时;要添加新行,程序只会在调试模式下没有任何警告的情况下崩溃。
在Winforms中,这可以通过使用Invoke函数轻松解决,但我不知道如何在WPF中解决这个问题。
答案 0 :(得分:157)
您可以使用
Dispatcher.Invoke(Delegate, object[])
在Application
(或任何UIElement
)调度员上。
您可以使用它,例如:
Application.Current.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));
或
someControl.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));
答案 1 :(得分:36)
最好的方法是从UI线程中获取SynchronizationContext
并使用它。此类将对其他线程的编组调用抽象化,并使测试更容易(与直接使用WPF的Dispatcher
相反)。例如:
class MyViewModel
{
private readonly SynchronizationContext _syncContext;
public MyViewModel()
{
// we assume this ctor is called from the UI thread!
_syncContext = SynchronizationContext.Current;
}
// ...
private void watcher_Changed(object sender, FileSystemEventArgs e)
{
_syncContext.Post(o => DGAddRow(crp.Protocol, ft), null);
}
}
答案 2 :(得分:4)
使用[Dispatcher.Invoke(DispatcherPriority, Delegate)]从其他线程或后台更改用户界面。
第1步。使用以下命名空间
using System.Windows;
using System.Threading;
using System.Windows.Threading;
第2步。将以下行放在需要更新UI的位置
Application.Current.Dispatcher.Invoke(DispatcherPriority.Background, new ThreadStart(delegate
{
//Update UI here
}));
<强>语法强>
[BrowsableAttribute(false)] public object Invoke( DispatcherPriority priority, Delegate method )
<强>参数强>
priority
输入:
System.Windows.Threading.DispatcherPriority
相对于其他待处理操作的优先级 Dispatcher事件队列,调用指定的方法。
method
输入:
System.Delegate
一个不带参数的方法的委托,它被推送到 Dispatcher事件队列。
返回值
输入:
System.Object
被调用的委托的返回值,如果是,则返回null 代表没有回报价值。
版本信息
自.NET Framework 3.0起可用