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

任何人都可以帮助我,我有一个问题我正在尝试通过线程池让这个代码在后台工作,但我似乎无法让它工作,我一直得到这个错误:

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"; } } } } } 

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

您可以通过执行此操作来访问controll

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

您无法从单独的线程访问该控件,它必须来自创建控件的同一线程。

这种扩展方法也解决了这个问题。

 ///  /// Allows thread safe updates of UI components ///  public static void InvokeEx(this T @this, Action action) where T : ISynchronizeInvoke { if (@this.InvokeRequired) { @this.Invoke(action, new object[] { @this }); } else { action(@this); } } 

在您的工作线程中使用如下

 InvokeEx(x => x.MyControl.Text = "foo"); 

您还可以使用以下语法通过调用来访问控件,并使用Action委托:

  Invoke((Action)(() => { var myVar = SomeWinFormControl.Property; })); 

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

您可以这样做,因为从UI线程以外的访问控件需要调用。

当您启动时(我假设您使用BackgroundWorker),将文本框中的url作为参数传递给RunWorkerAsync(TxtServer.Text) ,然后:

 private void DoWork(object o, DoWorkEventArgs e) { string Url = (string) e.Argument; List tmpList = new List; 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转换为列表并将其添加到您的控件中。