2

我正在使用 BackgroundWorker 定期检查硬件开关。由于它是通过慢速 RS485 网络连接的,我不得不延迟下一次状态更新。在切换状态更改时,我想更新 OK/nOK 图片框。这被实现为一个绿色的 OK pictureBox 在一个 noOK pictureBox 上。这里没有做任何真正的工作。

为了可扩展性,我决定使用 Backgroundworker。最后我想要一个隐藏的工人,

  1. 全局提供三个开关的状态和
  2. StatusChange 图片框的更新。

问题描述 BackgroundWorker 启动后,按预期工作。但是 GUI 冻结。

我尝试了什么?MSDN BackgroundWorker Class Note 1说 , 应该通过 ProgressChanged 更新 GUI。我试图通过 Worker_Switch.ReportProgress(fakeProgress++) 引发此事件但失败了。PictureBox 不再更新。

设计师的片段

this.Worker_Switch = new System.ComponentModel.BackgroundWorker();
// 
// Worker_Switch
// 
this.Worker_Switch.WorkerSupportsCancellation = true;
this.Worker_Switch.DoWork += new System.ComponentModel.DoWorkEventHandler(this.Worker_Switch_DoWork);

主窗体的片段

delegate void SetEventCallback(object sender, DoWorkEventArgs e);   // Threadsafe calls for DoWork

private void btnBackgroundworker_Click(object sender, EventArgs e)
{
    if (!Worker_Switch.IsBusy)
    {
        Worker_Switch.RunWorkerAsync();                              
    }
}

private void Worker_Switch_DoWork(object sender, DoWorkEventArgs e)
{
    // Worker Thread has no permission to change PictureBox "pictureBoxSwitchrightOK"
        // Therefore this method calls itsself in the MainThread, if necessary.
    while (!Worker_Switch.CancellationPending)
    {
      if (this.pictureBoxSwitchrightOK.InvokeRequired)              // Worker Thread
    {
            System.Threading.Thread.Sleep(400);
        SetEventCallback myCall = new SetEventCallback(Worker_Switch_DoWork);
        this.Invoke(myCall, new object[] { sender, e });
    }
    else                                                            // Main Thread
    {
        // Turns OK Picture Box invisible, if nOk State (Switch pushed)
        pictureBoxSwitchrightOK.Visible = SwitchOK("right");  // true: OK (green)
        this.Refresh();

    }
}

private bool SwitchOK(string rightOrLeft)               // select one of the switches
{ (...)}                                    // gets hardware switch status

编辑:特别感谢 laszlokiss88(3 种可能性)和 JMK(为了简单起见,使用System.Windows.Forms工具箱中的 Timer)

Toolbox 中的这种替代方法也有效:

this.timer_Switch.Enabled = true;
    this.timer_Switch.Interval = 400;
    this.timer_Switch.Tick += new System.EventHandler(this.timer_Switch_Tick);

    private void timer_Switch_Tick(object sender, EventArgs e)
{
    motorSwitchControl.Init();        // globally available Switch status                                               
    SwitchRight = SwitchOK("right");
    SwitchRightOK.Visible = SwitchRight; 

    SwitchLeft = SwitchOK("left");    // globally available Switch status
    SwitchLeftOK.Visible = SwitchLeft;  
    SwitchAllOK = SwitchRight & SwitchLeft;
    this.Refresh();
}

a) Sleep() 实际上发生在工作线程中是否正确?- 没有主线程

b) 如果我在 DoWork 中操作用户界面对象,会出现什么问题?(与 MSDN 注释相反)- 在主线程中工作?

c) 定期更新 PictureBox 的正确方法是什么?DoWork、ProgressChanged、RunWorkerCompleted...?- 来自 laszlokiss88 答案的三种可能性。

4

2 回答 2

5

您可以通过 Dispatcher 或 Control.Begininvoke(winforms) 从 DoWork 事件更新 UI,也可以通过 BackgroundWorker 的 ProgressChanged 事件进行更新:

    public MainWindow()
    {
        InitializeComponent();

        var bw = new BackgroundWorker();
        bw.WorkerReportsProgress = true;
        bw.DoWork += new DoWorkEventHandler(bw_DoWork);
        bw.ProgressChanged += new ProgressChangedEventHandler(bw_ProgressChanged);
        bw.RunWorkerAsync();
    }

    void bw_ProgressChanged(object sender, ProgressChangedEventArgs e)
    {
        // You are in the main thread
        // Update the UI here
        string data = (string)e.UserState;
    }

    void bw_DoWork(object sender, DoWorkEventArgs e)
    {
        // You are in a worker thread
        (sender as BackgroundWorker).ReportProgress(0, "right");
    }
于 2012-09-25T11:10:15.137 回答
1

首先,您几乎不需要让活动的背景广告进入睡眠状态。我也不确定你为什么要以这种方式构造/定义委托,尝试一些类似的东西

public delegate void UpdatePictureBox();
myDelegate = new UpdatePictureBox(UpdatePictureboxMethod);

那么你有一个方法UpdatePictureBoxMethod

private void UpdatePictureBoxMethod()
{
    this.pictureBox1.Image = Properties.Resources.SomeImage;
}

或类似的东西,您在其中传递要更新的图像。

或者,您可以使用该(bgWorker as BackgroundWorker).ReportProgress(progress, object);方法。所以从你调用的后台线程

(bgWorker as BackgroundWorker).ReportProgress(progressBarValue, infoBall);

这里的课程IfoBall将保存您所有的重要信息

class InfoBall
{
    public int nProgressBar { get; set; } 
    public int nMaxProgressBar { get; set; }
    public Image image { get; set; }
}

然后您可以将此对象传递回 UI 线程并进行更新

void bgWorker_ProgressChanged(object sender, ProgressChangedEventArgs e)
{
    // On UI thread.
    InfoBall someBall = (InfoBall)e.UserState;
    this.pictureBox1.Image = someBall.image;
    // etc...
}

我希望这有帮助。

于 2012-09-25T11:19:08.620 回答