在WPF中安全地访问UI(主)线程


95

我有一个应用程序,该应用程序每次以以下方式更新我正在监视的日志文件(附加新文本)时都会更新我的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.Add(ds); 要添加新行,程序会在调试模式下崩溃而没有任何警告。

在Winforms中,可以通过使用Invoke函数轻松解决此问题,但是我不确定如何在WPF中进行此操作。

Answers:


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 2012年

您可以为此使用任何UIElement,因为每个UIElement都具有“ Dispatcher”属性。
Wolfgang Ziegler

1
@ l46kok这可能有不同的原因(控制台应用程序,从winforms托管等)。正如@WolfgangZiegler所说,您可以使用任何UIElement。我通常用Application.Current它,因为它对我来说看起来更干净。
Botz3000

@ Botz3000我想我也在这里发生了一些比赛状况问题。附加上面给出的代码后,当我进入调试模式并手动执行逐步操作时,代码可以完美运行,但是当我在不调试的情况下运行应用程序时,代码将崩溃。我不确定要锁定在哪里会导致问题。
l46kok 2012年

1
@ l46kok如果您认为这是一个僵局,也可以致电Dispatcher.BeginInvoke。该方法只是将委托排队等待执行。
Botz3000

50

最好的解决方法是SynchronizationContext从UI线程获取并使用它。此类抽象化了对其他线程的编组调用,并使测试更加容易(与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);
    }
}

非常感谢!每次调用它时,已接受的解决方案便开始挂起,但这是可行的。
2013年

从包含视图模型但没有“真实” WPF的程序集(即类库)中调用时,它也可以工作。
2015年

这是一个非常有用的技巧,特别是当您有一个非wpf组件且其线程要编组操作时。当然,另一种方式是使用TPL延续
2015年

我一开始不理解,但是对我有用。(应该指出DGAddRow是私有方法)
Tim Davis

5

使用[Dispatcher.Invoke(DispatcherPriority,Delegate)]可以从另一个线程或后台更改UI。

步骤1。使用以下名称空间

using System.Windows;
using System.Threading;
using System.Windows.Threading;

第二步。将以下行放在需要更新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起可用

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.