WPFで安全にUI(メイン)スレッドにアクセスする


95

次の方法で、監視しているログファイルが更新されるたびに(新しいテキストが追加されて)データグリッドを更新するアプリケーションがあります。

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でこれをどのように行うかわかりません。

回答:


199

使用できます

Dispatcher.Invoke(Delegate, object[])

Applicationの(または任意UIElementの)ディスパッチャ。

たとえば、次のように使用できます。

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

または

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

上記のアプローチでは、ラインの実行時にApplication.Currentがnullであるためエラーが発生していました。なぜこれが当てはまるのでしょうか?
l46kok

すべてのUIElementには「Dispatcher」プロパティがあるため、そのために任意のUIElementを使用できます。
Wolfgang Ziegler、

1
@ l46kokこれにはさまざまな理由があります(コンソールアプリ、winformsからのホスティングなど)。@WolfgangZieglerが言ったように、任意のUIElementを使用できます。Application.Current見た目が綺麗なので、いつも使っています。
Botz3000 2012

@ Botz3000ここでも、競合状態の問題が発生していると思います。上記のコードを追加した後、デバッグモードで手動でステップオーバーするとコードは完全に機能しますが、デバッグなしでアプリケーションを実行するとコードがクラッシュします。ここで何をロックすれば問題が発生するかわかりません。
l46kok

1
@ l46kokデッドロックだと思われる場合は、を呼び出すこともできますDispatcher.BeginInvoke。そのメソッドは、デリゲートを実行のためにキューに入れるだけです。
Botz3000 2012

50

これに取り組む最良の方法SynchronizationContextは、UIスレッドからを取得して使用することです。このクラスは、他のスレッドへのマーシャリング呼び出しを抽象化し、(WPFをDispatcher直接使用する場合とは対照的に)テストを容易にします。例えば:

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

どうもありがとう!受け入れられたソリューションは、呼び出されるたびにハングし始めましたが、これは機能します。
Dov 2013年

また、ビューモデルを含むが「実際の」WPFを含まないアセンブリから呼び出された場合にも機能します。つまり、クラスライブラリです。
Onur

これは、特にアクションをマーシャリングしたいスレッドを持つ非wpfコンポーネントがある場合に、非常に便利なヒントです。もちろん、それを行う別の方法は、TPLの継続を使用することです
2015年

最初は理解できませんでしたが、うまくいきました。(DGAddRowはプライベートメソッドであることを指摘する必要があります)
Tim Davis

5

[Dispatcher.Invoke(DispatcherPriority、Delegate)]を使用して、UIを別のスレッドまたはバックグラウンドから変更します。

ステップ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

Dispatcherイベントキュー内の他の保留中の操作に対する相対的な優先順位で、指定されたメソッドが呼び出されます。

method

タイプ: System.Delegate

引数を取らないメソッドへのデリゲート。これはDispatcherイベントキューにプッシュされます。

戻り値

タイプ: System.Object

呼び出されているデリゲートからの戻り値。デリゲートに戻り値がない場合はnull。

バージョン情報

.NET Framework 3.0以降で使用可能

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.