从另一个线程获取控件的属性
本文关键字:属性 控件 获取 另一个 线程 | 更新日期: 2023-09-27 18:34:11
我想从我的窗体中的BackgroundWorker
中获取控件的属性:
foreach (ListViewItem i in ListView.CheckedItems) { //error: Cross-thread operation not valid: Control 'ListView' accessed from a thread other than the thread it was created on.
//do something with i
}
任何人都可以建议最简单,最简单的方法来做到这一点吗?
让我再试一试...
1.) 将列表视图拖到窗体上
2.) 将后台工作者拖到窗体上
3.) 创建一个方法,循环访问列表视图项集合
private void LoopThroughListItems()
{
foreach (ListViewItem i in listView1.CheckedItems)
DoSomething();
}
4.) 添加代码以在 BackgroundWorker 的 DoWork 事件中调用 LoopThroughListItems()
private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
{
LoopThroughListItems();
}
5.) 在表单加载中 - 在主线程上执行代码(它有效),然后在后台工作线程上执行代码(它失败)
private void Form1_Load(object sender, EventArgs e)
{
// Try it on the UI Thread - It works
LoopThroughListItems();
// Try it on a Background Thread - It fails
backgroundWorker1.RunWorkerAsync();
}
6.) 修改代码以使用 IsInvokeRequired/Invoke
private void LoopThroughListItems()
{
// InvokeRequired == True when executed by non-UI thread
if (listView1.InvokeRequired)
{
// This will re-call LoopThroughListItems - on the UI Thread
listView1.Invoke(new Action(LoopThroughListItems));
return;
}
foreach (ListViewItem i in listView1.CheckedItems)
DoSomething();
}
7.) 再次运行应用程序 - 现在它可以在 UI 线程和非 UI 线程上运行。
这就解决了问题。 检查 IsInvokeRequired/Invoking 是一种常见的模式,你会习惯很多(这就是为什么它包含在所有控件中)。 如果你到处都在做,你可以做一些聪明的事情并将其全部打包 - 如下所述:自动执行调用所需的代码模式
尝试这样的事情:
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Windows.Forms;
namespace WindowsFormsApplication1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
private void OnClick(object sender, EventArgs e)
{
backgroundWorker1.RunWorkerAsync();
}
private void OnDoWork(object sender, DoWorkEventArgs e)
{
foreach (ListViewItem i in GetItems(listView1))
{
DoSomething(i);
}
}
private IEnumerable<ListViewItem> GetItems(ListView listView)
{
if (InvokeRequired)
{
var func = new Func<ListView, IEnumerable<ListViewItem>>(GetItems);
return (IEnumerable<ListViewItem>)Invoke(func, new[] { listView });
}
// Create a defensive copy to avoid iterating outsite UI thread
return listView.CheckedItems.OfType<ListViewItem>().ToList();
}
private void DoSomething(ListViewItem item)
{
if (InvokeRequired)
{
var action = new Action<ListViewItem>(DoSomething);
Invoke(action, new[] { item });
return;
}
// Do whatever you want with i
item.Checked = false;
}
}
}
但是,你的问题真的很笼统。如果您分享更多详细信息,也许会有一个更简单或更好的解决方案。