Bezpieczny dostęp do wątku interfejsu użytkownika (głównego) w WPF

99

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.

l46kok
źródło

Odpowiedzi:

203

Możesz użyć

Dispatcher.Invoke(Delegate, object[])

na wysyłającym Application(lub innym UIElement).

Możesz go użyć na przykład w ten sposób:

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

lub

someControl.Dispatcher.Invoke(new Action(() => { /* Your code here */ }));
Botz3000
źródło
Powyższe podejście dawało błąd, ponieważ Application.Current ma wartość NULL w momencie uruchamiania linii. Dlaczego tak się dzieje?
l46kok
Możesz użyć do tego dowolnego elementu UIElement, ponieważ każdy element UIElement ma właściwość „Dispatcher”.
Wolfgang Ziegler,
1
@ l46kok Może to mieć różne przyczyny (aplikacja konsolowa, hosting z winForms itp.). Jak powiedział @WolfgangZiegler, możesz użyć do tego dowolnego elementu UIElement. Zwykle go używam Application.Current, ponieważ wydaje mi się czystszy.
Botz3000,
@ Botz3000 Myślę, że mam tu również problem z warunkami wyścigu. Po dołączeniu podanego powyżej kodu, kod działa idealnie, gdy wchodzę w tryb debugowania i ręcznie wykonuję stepover, ale ulega awarii, gdy uruchamiam aplikację bez debugowania. Nie jestem pewien, co tutaj zablokować, co powoduje problem.
l46kok
1
@ l46kok Jeśli uważasz, że to impas, możesz również zadzwonić Dispatcher.BeginInvoke. Ta metoda po prostu kolejkuje delegata do wykonania.
Botz3000,
51

Najlepszym sposobem byłoby pobranie SynchronizationContextz 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 do Dispatcherbezpoś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);
    }
}
Eli Arbel
źródło
Wielkie dzięki! Zaakceptowane rozwiązanie zaczęło się zawieszać przy każdym wywołaniu, ale to działa.
Dov
Działa również, gdy jest wywoływana z zestawu zawierającego model widoku, ale nie ma „prawdziwego” WPF, tj. Jest biblioteką klas.
Onur
Jest to bardzo przydatna wskazówka, szczególnie gdy masz składnik inny niż wpf z wątkiem, do którego chcesz organizować akcje. oczywiście innym sposobem byłoby skorzystanie z kontynuacji OC
MaYaN
Na początku tego nie rozumiałem, ale zadziałało dla mnie… niezłe. (Należy zaznaczyć, że DGAddRow to metoda prywatna)
Tim Davis
5

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
}));

Składnia

[BrowsableAttribute(false)]
public object Invoke(
  DispatcherPriority priority,
  Delegate method
)

Parametry

priority

Rodzaj: System.Windows.Threading.DispatcherPriority

Priorytet w stosunku do innych oczekujących operacji w kolejce zdarzeń Dispatcher, wywoływana jest określona metoda.

method

Rodzaj: System.Delegate

Delegat do metody, która nie przyjmuje argumentów, która jest wypychana do kolejki zdarzeń Dispatcher.

Wartość zwracana

Rodzaj: System.Object

Wartość zwracana przez wywoływanego delegata lub null, jeśli delegat nie ma wartości zwracanej.

Informacje o wersji

Dostępne od .NET Framework 3.0

Vineet Choudhary
źródło