What's wrong with my cross-thread call in Windows Forms?

后端 未结 7 1915
南笙
南笙 2021-01-04 22:32

I encounter a problem with a Windows Forms application.

A form must be displayed from another thread. So in the form class, I have the following code:



        
相关标签:
7条回答
  • 2021-01-04 23:09

    Most likely the handle of the control is not created yet, in which case Control.InvokeRequired returns false.

    Check the Control.IsHandleCreated property to see if this is the case.

    0 讨论(0)
  • 2021-01-04 23:09

    I also think SLaks is correct. From msdn (http://msdn.microsoft.com/en-us/library/system.windows.forms.control.invokerequired.aspx):

    If no appropriate handle can be found, the InvokeRequired method returns false.

    If its possible in your case, I would try to combine the creating and showing the control in a single method, i.e.:

    public DisplayDialog static Show()
    {
      var result = new DisplayDialog; //possibly cache instance of the dialog if needed, but this could be tricky
      result.ShowDialog(); 
      return result;
    }
    

    you can call Show from a different thread

    0 讨论(0)
  • 2021-01-04 23:11

    Try this one:

    private delegate void DisplayDialogCallback();
    
    public void DisplayDialog()
    {
        if (this.InvokeRequired)
        {
            this.Invoke(new DisplayDialogCallback(DisplayDialog));
        }
        else
        {
            if (this.Handle != (IntPtr)0) // you can also use: this.IsHandleCreated
            {
                this.ShowDialog();
    
                if (this.CanFocus)
                {
                    this.Focus();
                }
            }
            else
            {
                // Handle the error
            }
        }
    }
    

    Please note that InvokeRequired returns

    true if the control's Handle was created on a different thread than the calling thread (indicating that you must make calls to the control through an invoke method); otherwise, false.

    and therefore, if the control has not been created, the return value will be false!

    0 讨论(0)
  • 2021-01-04 23:11

    You could always try testing against a different control.

    For example, you could access Application.Forms collections

    public Control GetControlToInvokeAgainst()
    {
        if(Application.Forms.Count > 0)
        {
            return Application.Forms[0];
        }
        return null;
    }
    

    Then in your DisplayDialog() method, call the GetControlToInvokeAgainst() and test for null before trying to perform an invokerequired call.

    0 讨论(0)
  • 2021-01-04 23:18

    I believe what is happening here is that this code is being run before the Form is ever shown.

    When a Form is created in .Net it does not immediately gain affinity for a particular thread. Only when certain operations are performed like showing it or grabbing the handle does it gain affinity. Before that happens it's hard for InvokeRequired to function correctly.

    In this particular case no affinity is established and no parent control exists so InvokeRequired returns false since it can't determine the original thread.

    The way to fix this is to establish affinity for your control when it's created on the UI thread. The best way to do this is just to ask the control for it's handle property.

    var notUsed = control.Handle;
    
    0 讨论(0)
  • 2021-01-04 23:21

    You are likely getting to this code before the form has been shown and therefore the window handle has not been created.

    You can add this code before your code and all should be good:

    if (! this.IsHandleCreated)
       this.CreateHandle();
    

    Edit: There's another problem with your code. Once the form is displayed, you cannot call ShowDialog() again. You will get an invalid operation exception. You may want to modify this method as others have proposed.

    You might be better served calling the ShowDialog() directly from the calling class and have another method for BringToFront() or something like that...

    0 讨论(0)
提交回复
热议问题