Ich habe eine Anwendung, die mein Datagrid jedes Mal aktualisiert, wenn eine von mir beobachtete Protokolldatei auf folgende Weise aktualisiert (mit neuem Text angehängt) wird:
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);
}
Wenn das Ereignis für den FileWatcher ausgelöst wird, weil es einen separaten Thread erstellt, wenn ich versuche, dataGridRows.Add (ds) auszuführen; Um die neue Zeile hinzuzufügen, stürzt das Programm nur ab, ohne dass im Debug-Modus eine Warnung ausgegeben wird.
In Winforms wurde dies leicht mithilfe der Invoke-Funktion gelöst, aber ich bin mir nicht sicher, wie ich dies in WPF tun soll.
Application.Current
es normalerweise , da es für mich sauberer aussieht.Dispatcher.BeginInvoke
. Diese Methode stellt den Delegaten nur zur Ausführung in die Warteschlange.Der beste Weg, dies zu tun, wäre, einen
SynchronizationContext
aus dem UI-Thread zu holen und ihn zu verwenden. Diese Klasse abstrahiert Marshalling-Aufrufe an andere Threads und erleichtert das Testen (im Gegensatz zurDispatcher
direkten Verwendung von WPFs ). Beispielsweise:quelle
Verwenden Sie [Dispatcher.Invoke (DispatcherPriority, Delegate)] , um die Benutzeroberfläche von einem anderen Thread oder vom Hintergrund aus zu ändern.
Schritt 1 . Verwenden Sie die folgenden Namespaces
Schritt 2 . Fügen Sie die folgende Zeile ein, in der Sie die Benutzeroberfläche aktualisieren müssen
quelle