Mam aplikację, która aktualizuje mój datagrid za każdym razem, gdy plik dziennika, który oglądam, zostanie zaktualizowany (dołączony do nowego tekstu) w następujący sposób:
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);
}
Gdy zdarzenie jest zgłaszane dla FileWatcher, ponieważ tworzy oddzielny wątek, gdy próbuję uruchomić dataGridRows.Add (ds); aby dodać nowy wiersz, program po prostu zawiesza się bez żadnego ostrzeżenia w trybie debugowania.
W Winforms problem ten można było łatwo rozwiązać, wykorzystując funkcję Invoke, ale nie jestem pewien, jak to zrobić w WPF.
Application.Current
, ponieważ wydaje mi się czystszy.Dispatcher.BeginInvoke
. Ta metoda po prostu kolejkuje delegata do wykonania.Najlepszym sposobem byłoby pobranie
SynchronizationContext
z wątku interfejsu użytkownika i użycie go. Ta klasa abstraktuje wywołania do innych wątków i ułatwia testowanie (w przeciwieństwie doDispatcher
bezpośredniego korzystania z WPF ). Na przykład: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); } }
źródło
Użyj [Dispatcher.Invoke (DispatcherPriority, Delegate)], aby zmienić interfejs użytkownika z innego wątku lub z tła.
Krok 1 . Użyj następujących przestrzeni nazw
using System.Windows; using System.Threading; using System.Windows.Threading;
Krok 2 . Umieść następujący wiersz w miejscu, w którym chcesz zaktualizować interfejs użytkownika
Application.Current.Dispatcher.Invoke(DispatcherPriority.Background, new ThreadStart(delegate { //Update UI here }));
źródło