Безопасный доступ к потоку UI (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.Добавить (ds); чтобы добавить новую строку, программа просто аварийно завершает работу без предупреждения, данного в режиме отладки.
в Winforms это было легко решено с помощью функции Invoke, но я не знаете, как это сделать в WPF.
3 ответа:
можно использовать
Dispatcher.Invoke(Delegate, object[])на
Application'S (или любойUIElementС) диспетчер.вы можете использовать его, например, так:
Application.Current.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));или
someControl.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));
лучший способ сделать это было бы получить
SynchronizationContextиз потока пользовательского интерфейса и использовать его. Этот класс абстрагирует вызовы маршалинга для других потоков и упрощает тестирование (в отличие от использования WPFDispatcherнапрямую). Например: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); } }
использовать [диспетчером.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приоритет по отношению к другим незавершенным операциям в системе Очередь событий диспетчера, вызывается указанный метод.
methodтип:
System.Delegateделегат к методу, который не принимает аргументов, который нажимается на очередь событий диспетчера.
Возвращаемое Значение
тип:
System.Objectвозвращаемое значение делегат, который вызывается, или null, если делегат не имеет возвращаемого значения.
Информация О Версии
доступно с .NET Framework 3.0