跨线程操作在Windows窗体中无效

7

有人可以帮我吗?我遇到了一个问题。我正在尝试通过线程池在后台运行此代码,但似乎无法使其正常运行,我一直得到以下错误:

Cross-thread operation not valid: Control 'ListBox3' accessed
from a thread other than the thread it was created on. 

这是我正在使用的代码:

private void DoWork(object o)
{
    var list = ListBox3;

    var request = createRequest(TxtServer.Text, WebRequestMethods.Ftp.ListDirectory);

    using (var response = (FtpWebResponse)request.GetResponse())
    {
        using (var stream = response.GetResponseStream())
        {
            using (var reader = new StreamReader(stream, true))
            {
                while (!reader.EndOfStream)
                {
                    list.Items.Add(reader.ReadLine());
                    ResultLabel.Text = "Connected";
                }
            }
        }
    }
}
7个回答

11

你可以通过这样做来访问控制

 Invoke(new Action(() => {Foo.Text="Hi";}));

6

您需要调用委托来更新列表。请参阅此MSDN示例


这是正确的答案 - 如果需要,使用委托进行调用。请查看提供链接中的“对 Windows 窗体控件的线程安全调用”部分。 - Algirdas

3
您还可以使用以下语法通过调用来访问控件,其中包含一个Action委托:
 Invoke((Action)(() =>
 {
      var myVar = SomeWinFormControl.Property;
 }));

3
这个扩展方法也能解决问题。
/// <summary>
/// Allows thread safe updates of UI components
/// </summary>
public static void InvokeEx<T>(this T @this, Action<T> action) where T : ISynchronizeInvoke
{
    if (@this.InvokeRequired)
    {
        @this.Invoke(action, new object[] { @this });
    }
    else
    {
        action(@this);
    }
}

在您的工作线程中使用如下:
InvokeEx(x => x.MyControl.Text = "foo");

2

你无法从一个单独的线程访问控件,必须在创建该控件的相同线程中访问。


1
我假设DoWork在另一个线程上启动。该代码访问ListBox3,它是一个GUI控件。.NET限制对GUI控件的访问仅限于创建它们的线程。

你需要将对控件的访问调度回UI线程。你是如何调用“DoWork”的? - Brian Rasmussen
当按钮被点击时,它会执行以下命令:ThreadPool.QueueUserWorkItem(new WaitCallback(DoWork)); - Terrii

1

你可以这样做,因为从非 UI 线程访问控件需要使用 invoke。

当你开始执行(我假设你使用 BackgroundWorker),将文本框中的 URL 作为参数传递给 RunWorkerAsync(TxtServer.Text),然后:

private void DoWork(object o, DoWorkEventArgs e)
{
    string Url = (string) e.Argument;

    List<of string> tmpList = new List<of string>;

    var request = createRequest(url, WebRequestMethods.Ftp.ListDirectory);

    using (var response = (FtpWebResponse)request.GetResponse())
    {
        using (var stream = response.GetResponseStream())
        {
            using (var reader = new StreamReader(stream, true))
            {
                while (!reader.EndOfStream)
                {
                    list.Add(reader.ReadLine());
                    //ResultLabel.Text = "Connected";
                    //use reportprogress() instead
                }
            }
        }
    }
    e.result = tmpList;
}

然后在您的Completed事件中将e.result转换为列表并将其添加到您的控件中。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接