C# - 如何从另一个类的线程更新主UI

时间:2015-04-20 04:39:16

标签: c# wpf multithreading events delegates

我实际上正在学习(艰难的方式)c#并且一直在与问题作斗争:

我正在用WPF(dotNet 4.0)编写我的第一个c#应用程序。当我单击一个按钮时,使用BackgroundWorker线程并从外部类调用方法,这样我的UI就不会冻结 - >我的方法按预期运行。

然后我尝试从外部类更新ListView控件以获得某种进展(文本),但我很难失败。

我知道我需要使用委托和调度程序来更新我的控件。

我尝试使用此处提供的解决方案How to update UI from another thread running in another class。 (由于我的低代表,我不能评论它)我错过了这个难题的一些部分。

YourEventArgs(status)指的是什么?当我的方法在BGW内部运行时,我只是没有办法触发事件并将内容传回我的UI。

到目前为止,我有这段代码(从答案中更新):

namespace AppMain
{
    public partial class MainWindow
    {
        BackgroundWorker AppWorker = new BackgroundWorker();

        public MainWindow()
        {
            InitializeComponent();

            AppWorker.WorkerSupportsCancellation = true;
            AppWorker.DoWork += AppWorker_DoWork;
            AppWorker.RunWorkerCompleted += AppWorker_RunWorkerCompleted;
            }

        private void btnLoad_Click(object sender, RoutedEventArgs e)
        {
            lstTest.Items.Add("Processing data...");
            AppWorker.RunWorkerAsync();
        }

        public void AppWorker_DoWork(object sender, System.ComponentModel.DoWorkEventArgs e)
        {
            SetXmlData xml = new SetXmlData();
            xml.ProgressUpdate += (s, evt) =>
            {
                Dispatcher.BeginInvoke((Action)(() =>
                {
                    lstTest.Items.Add("this is a test : " + evt.myData); //how to retrieve the myData property from evt ?
                }));
            };
            xml.FlushData();
        }

        public void AppWorker_RunWorkerCompleted(object sender, System.ComponentModel.RunWorkerCompletedEventArgs e)
        {
            if (!(e.Cancelled))
            {
                lstTest.Items.Add("Done");
            }
            else
            {
                MessageBox.Show("Cancelled");
            }
        }
}
}

SetXmlData.cs

namespace AppMain
{
    public class SetXmlData
    {
    public event EventHandler ProgressUpdate;

        //update method
        public void update(object input)
        {
        if (ProgressUpdate != null)
        ProgressUpdate(this, new YourEventArgs { myData = (string)input });
        }

        //calculation method
        public void FlushData()
        {
            MessageBox.Show("this is a test !");
            update("test");
        }
    }

    public class YourEventArgs : EventArgs
    {
        public string myData { get; set; }
    }
}

感谢您的帮助。

2 个答案:

答案 0 :(得分:0)

您只需从ProgressUpdate方法调用FlushData()事件即可。

只需致电:

If (ProgressUpdate !=null )
{
    ProgressUpdate(this,new YourEventArgs())
}

thisevent源自的源实例。

您可以通过继承EventArgs类来创建YourEventArgs。

  public class YourEventArgs : EventArgs
  {
     //Put any property that you want to pass back to UI here.
  }

在用户界面中引发event

RaiseEvent.ProgressUpdate += (s, e) =>
        {
            Dispatcher.BeginInvoke((Action)(() => 
                                        { 
                                            lstTest.Items.Add("this is a test : "); 
                                            //Add items to your UI control here...
                                        }));
        };

e将是YourEventArgs类型。

另外,您不应该从不同的线程(例如示例中的后台工作线程)触摸UI线程。由于您的event-handler已经执行了Dispatcher.BeginInvoke,这是安全的。

此外,您的ProgressUpdate事件应该在您的类SetXmlData中。

答案 1 :(得分:0)

尝试get; set;例如:

Form1中:

public partial class Form1 : Form
{
    static public string gettext { get; set; }

    public Form1()
    {
        InitializeComponent();
    }

    private void Form1_Load(object sender, EventArgs e)
    {
        Class1.send(); //call to class function
        textBox1.Text = gettext; //items.add(gettext)
    }
}

的Class1:

    class Class1
{
    static public void send()
    {
        Form1.gettext = "Marko"; //Set gettext to string "Marko"

    }
}