it-swarm.com.ru

Безопасный доступ к пользовательскому (основному) потоку в WPF

У меня есть приложение, которое обновляет мою сетку данных каждый раз, когда файл журнала, который я наблюдаю, обновляется (с добавлением нового текста) следующим образом:

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.

73
l46kok

Ты можешь использовать

Dispatcher.Invoke(Delegate, object[])

на диспетчера Application (или любого UIElement).

Вы можете использовать его, например, так:

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

или же

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

Лучший способ сделать это - получить SynchronizationContext из потока пользовательского интерфейса и использовать его. Этот класс абстрагирует вызовы маршалинга для других потоков и упрощает тестирование (в отличие от непосредственного использования Dispatcher в WPF). Например:

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);
    }
}
36
Eli Arbel

Используйте [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

Приоритет, относительно других ожидающих операций в очереди событий Dispatcher, вызывается указанным методом.

method

Тип: System.Delegate

Делегат метода, который не принимает аргументов, который помещается в очередь событий Dispatcher.

Возвращаемое значение

Тип: System.Object

Возвращаемое значение из вызываемого делегата или нуль, если у делегата нет возвращаемого значения.

Информация о версии

Доступно с .NET Framework 3.0

4
Vineet Choudhary