WPF에서 안전하게 UI (메인) 스레드에 액세스


99

내가보고있는 로그 파일이 다음과 같은 방식으로 업데이트 (새 텍스트로 추가됨) 될 때마다 데이터 그리드를 업데이트하는 응용 프로그램이 있습니다.

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에서 어떻게해야할지 모르겠습니다.

답변:


203

당신이 사용할 수있는

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를 사용할 수 있습니다.
볼프강 지글러

1
@ l46kok 다른 이유 (콘솔 앱, winforms에서 호스팅 등)가있을 수 있습니다. @WolfgangZiegler가 말했듯이 모든 UIElement를 사용할 수 있습니다. 나는 Application.Current그것이 나에게 더 깨끗해 보이기 때문에 보통 그것을 사용 합니다.
Botz3000

@ Botz3000 나는 또한 여기에서 발생하는 경쟁 조건 문제가 있다고 생각합니다. 위에 주어진 코드를 추가 한 후 디버그 모드로 들어가 수동으로 스텝 오버를 수행하면 코드가 완벽하게 작동하지만 디버그없이 애플리케이션을 실행하면 코드가 충돌합니다. 문제를 일으키는 여기에서 무엇을 잠글 지 잘 모르겠습니다.
l46kok

1
@ l46kok 교착 상태라고 생각되면 Dispatcher.BeginInvoke. 이 메서드는 실행을 위해 대리자를 큐에 넣습니다.
Botz3000 2012

51

이를 해결하는 가장 좋은 방법 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

뷰 모델을 포함하지만 "실제"WPF가없는 어셈블리 (즉, 클래스 라이브러리)에서 호출 될 때도 작동합니다.
Onur

이것은 특히 작업을 마샬링하려는 스레드가있는 비 wpf 구성 요소가있는 경우 매우 유용한 팁입니다. 물론 다른 방법은 TPL 연속을 사용하는 것입니다
MaYaN

처음에는 이해하지 못했지만 저에게는 효과가있었습니다. (DGAddRow 아웃해야 점은 개인 방법은)
팀 데이비스

5

다른 스레드 또는 백그라운드에서 UI를 변경 하려면 [Dispatcher.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

Dispatcher 이벤트 큐에서 다른 보류중인 작업에 상대적인 우선 순위로 지정된 메서드가 호출됩니다.

method

유형: System.Delegate

Dispatcher 이벤트 큐로 푸시되는 인수를받지 않는 메서드에 대한 대리자입니다.

반환 값

유형: System.Object

호출되는 대리자의 반환 값 또는 대리자에 반환 값이없는 경우 null입니다.

버전 정보

.NET Framework 3.0부터 사용 가능

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.