跨线程操作在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转换为列表并将其添加到您的控件中。