การเข้าถึง UI (Main) เธรดอย่างปลอดภัยใน WPF


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

คำตอบ:


204

คุณสามารถใช้ได้

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 เป็นโมฆะในขณะที่รันบรรทัด ทำไมถึงเป็นเช่นนี้?
l46kok

คุณสามารถใช้ UIElement ใดก็ได้เนื่องจาก UIElement ทุกตัวมีคุณสมบัติ "Dispatcher"
Wolfgang Ziegler

1
@ l46kok ซึ่งอาจมีสาเหตุที่แตกต่างกัน (แอปคอนโซลการโฮสต์จาก winforms เป็นต้น) ดังที่ @WolfgangZiegler กล่าวไว้คุณสามารถใช้ UIElement ใด ๆ ก็ได้ ฉันมักจะใช้Application.Currentมันเพราะมันดูสะอาดกว่าสำหรับฉัน
Botz3000

@ Botz3000 ฉันคิดว่าฉันยังมีปัญหาสภาพการแข่งขันเกิดขึ้นที่นี่ หลังจากต่อท้ายรหัสที่ระบุข้างต้นโค้ดจะทำงานได้อย่างสมบูรณ์เมื่อฉันเข้าสู่โหมดดีบักและทำ stepovers ด้วยตนเอง แต่โค้ดจะขัดข้องเมื่อฉันเรียกใช้แอปพลิเคชันโดยไม่มีการดีบัก ฉันไม่แน่ใจว่าจะล็อคอะไรที่ทำให้เกิดปัญหา
l46kok

1
@ l46kok ถ้าคิดว่าเป็นทางตันก็โทรDispatcher.BeginInvoke. วิธีการนั้นจะจัดคิวผู้รับมอบสิทธิ์เพื่อดำเนินการ
Botz3000

52

วิธีที่ดีที่สุดคือการรับ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 เป็นวิธีการส่วนตัว)
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

ลำดับความสำคัญที่สัมพันธ์กับการดำเนินการอื่น ๆ ที่รอดำเนินการในคิวเหตุการณ์ของผู้มอบหมายงานวิธีการที่ระบุจะถูกเรียกใช้

method

ประเภท: System.Delegate

ผู้รับมอบสิทธิ์ให้กับเมธอดที่ไม่มีอาร์กิวเมนต์ซึ่งจะถูกส่งไปยังคิวเหตุการณ์ของผู้มอบหมายงาน

ส่งคืนค่า

ประเภท: System.Object

ค่าส่งคืนจากผู้รับมอบสิทธิ์ที่ถูกเรียกใช้หรือเป็นโมฆะถ้าผู้รับมอบสิทธิ์ไม่มีค่าส่งคืน

ข้อมูลเวอร์ชัน

พร้อมใช้งานตั้งแต่. NET Framework 3.0

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.