C# - 即使在调用之后,在不同的线程中添加UserControl也会导致异常

Sha*_*ssy 4 c# user-controls multithreading winforms

我在winfroms中构建了一个自定义UserControl,并通过辅助线程将其添加到Panel中.

我知道当通过辅助线程添加控件时,您需要调用主线程来执行它.所以我做了..但我仍然得到一个例外,说"跨线程操作无效:控制'pictureBoxImage'从一个线程访问,而不是它创建的线程."

我被卡住了,不知道是什么导致了这个因为我尝试通过在每个自定义UserControl方法上放置一个断点来调试它,但是它们中的任何一个都没有引发异常.

private void addControl(Control i_ControllToAdd, Control i_ParentControl)
    {
        if (i_ParentControl.InvokeRequired)
        {
            i_ParentControl.Invoke(new Action(() => addControl(i_ControllToAdd, i_ParentControl)));
            return;
        }

        i_ParentControl.Controls.Add(i_ControllToAdd);
    }
Run Code Online (Sandbox Code Playgroud)

这是自定义的UserControl类

public partial class FBPostUserControl : UserControl
    {
        private readonly string m_UserName = string.Empty;
        private readonly Image m_UserProfileImage = null;
        private readonly DateTime? m_DatePosted = null;
        private Image m_Image = null;
        private string m_PostBody = string.Empty;

    public string UserName
    {
        get { return m_UserName; }
    }

    public DateTime? DatePosted
    {
        get { return m_DatePosted; }
    }

    public Image Image
    {
        get { return m_Image; }
        set
        {
            if (value == null)
            {
                pictureBoxImage.Visible = false;
            }
            else
            {
                pictureBoxImage.Visible = true;
                pictureBoxImage.Image = value;
                updateImageSize();
            }
        }
    }

    private void updateImageSize()
    {
        if (pictureBoxImage.Image != null)
        {
            double ratio = pictureBoxImage.Image.Width / pictureBoxImage.Image.Height;
            pictureBoxImage.Height = (int)(pictureBoxImage.Width / ratio);
            pictureBoxImage.SizeMode = PictureBoxSizeMode.Zoom;
        }
    }

    public string PostBody
    {
        get { return m_PostBody; }
        set
        {
            if (string.IsNullOrWhiteSpace(value) == false)
            {
                labelPostBody.Visible = true;
                labelPostBody.Text = value;
            }
            else
            {
                labelPostBody.Visible = false;
            }
        }
    }

    public Image UserProfileImage
    {
        get { return m_UserProfileImage; }
    }

    public FBPostUserControl(string i_Name, Image i_ProfileImage, DateTime? i_PostDate)
    {
        InitializeComponent();
        m_UserName = i_Name;
        m_UserProfileImage = i_ProfileImage;
        m_DatePosted = i_PostDate;

        refreshHeader();
    }

    private void refreshHeader()
    {
        pictureBoxUserImage.Image = m_UserProfileImage;
        labelName.Text = m_UserName;

        if (labelDate != null)
        {
            labelDate.Text = m_DatePosted.ToString();
        }
        else
        {
            labelDate.Visible = false;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

Kam*_*ede 5

任何Control都应该在UI线程上创建,并且也可以从同一个线程访问.

它之所以给你异常,是因为你是ControlToAdd在一个线程(你的辅助线程)上创建你的,并使用Invoke语句将它添加到另一个线程(你的UI线程)的UI上.

要摆脱这种情况,请确保在一个线程上创建控件并在其创建的同一线程上访问它.

以下片段可能不是您想要的,但它只是为了让您了解它;

i_ParentControl.Invoke(new Action(() => addControl(new Control(), i_ParentControl)));
Run Code Online (Sandbox Code Playgroud)