在 WPF 中安全访问 UI (主)线程

我有一个应用程序,每当我正在观看的日志文件更新时,它都会以下列方式更新我的 datagrid (附加新文本) :

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);
}

当尝试运行 dataGridRows 时,为 FileWatcher 引发事件,因为它创建了一个单独的线程。添加(ds) ; 要添加新行,程序在调试模式下会在没有任何警告的情况下崩溃。

在 Winform 中,通过使用 Invoke 函数可以很容易地解决这个问题,但是我不确定如何在 WPF 中解决这个问题。

152588 次浏览

你可以用

Dispatcher.Invoke(Delegate, object[])

Application的(或任何 UIElement的)调度器上。

你可以这样使用它:

Application.Current.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));

或者

someControl.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));

最好的方法是从 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);
}
}

使用 [调度程序。调用(调度程序优先级,委托)]可以从另一个线程或后台更改 UI。

步骤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

控件中相对于其他挂起操作的优先级 事件队列,则调用指定的方法。

method

Type: System.Delegate

指向不接受参数的方法的委托,该方法被推送到 Dispatcher 事件队列。

返回值

类型: System.Object

调用的委托的返回值,如果 委托没有返回值。

版本资料

从.NET Framework 3.0开始可用