Доступ к интерфейсу пользовательского интерфейса (Main) безопасно в WPF
У меня есть приложение, которое обновляет мой 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);
}
Когда событие создается для FileWatcher, потому что он создает отдельный поток, когда я пытаюсь запустить dataGridRows.Add(ds); для добавления новой строки программа просто выходит из строя без предупреждения, заданного во время режима отладки.
В Winforms это легко решить, используя функцию Invoke, но я не уверен, как это сделать в WPF.
Ответы
Ответ 1
Вы можете использовать
Dispatcher.Invoke(Delegate, object[])
в диспетчере Application
(или любом UIElement
).
Вы можете использовать его, например, следующим образом:
Application.Current.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));
или
someControl.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));
Ответ 2
Лучший способ сделать это - получить 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);
}
}
Ответ 3
Используйте [Dispatcher.Invoke(DispatcherPriority, Delegate)], чтобы изменить пользовательский интерфейс из другого потока или из фона.
Шаг 1. Используйте следующие пространства имен
using System.Windows;
using System.Threading;
using System.Windows.Threading;
Шаг 2. Поместите следующую строку, где вам нужно обновить интерфейс
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
Тип: System.Delegate
Делегат метода, который не принимает аргументов, которые нажимаются на очередь событий диспетчера.
Возвращаемое значение
Тип: System.Object
Возвращаемое значение вызываемого делегата или null, если делегат не имеет возвращаемого значения.
Информация о версии
Доступно с .NET Framework 3.0