这是我第一次发布在StackOverflow上,所以请原谅下面的乱码格式。
为了防止在更新ListView时锁定窗体,可以使用下面我写的方法来解决此问题。
注意:如果您希望用超过20,000个项目填充ListView,则不应使用此方法。如果您需要向ListView添加20多个项目,请考虑以虚拟模式运行ListView。
public static async void PopulateListView<T>(ListView listView, Func<T, ListViewItem> func,
IEnumerable<T> objects, IProgress<int> progress) where T : class, new()
{
if (listView != null && listView.IsHandleCreated)
{
var conQue = new ConcurrentQueue<ListViewItem>();
// Clear the list view and refresh it
if (listView.InvokeRequired)
{
listView.BeginInvoke(new MethodInvoker(() =>
{
listView.BeginUpdate();
listView.Items.Clear();
listView.Refresh();
listView.EndUpdate();
}));
}
else
{
listView.BeginUpdate();
listView.Items.Clear();
listView.Refresh();
listView.EndUpdate();
}
// Loop over the objects and call the function to generate the list view items
if (objects != null)
{
int objTotalCount = objects.Count();
foreach (T obj in objects)
{
await Task.Run(() =>
{
ListViewItem item = func.Invoke(obj);
if (item != null)
conQue.Enqueue(item);
if (progress != null)
{
double dProgress = ((double)conQue.Count/objTotalCount) * 100.0;
if(dProgress > 0)
progress.Report(dProgress > int.MaxValue ? int.MaxValue : (int)dProgress);
}
});
}
// Perform a mass-add of all the list view items we created
if (listView.InvokeRequired)
{
listView.BeginInvoke(new MethodInvoker(() =>
{
listView.BeginUpdate();
listView.Items.AddRange(conQue.ToArray());
listView.Sort();
listView.EndUpdate();
}));
}
else
{
listView.BeginUpdate();
listView.Items.AddRange(conQue.ToArray());
listView.Sort();
listView.EndUpdate();
}
}
}
if (progress != null)
progress.Report(100);
}
您不必提供IProgress对象,只需使用null,该方法也可以正常工作。
下面是该方法的示例用法。
首先,定义一个包含ListViewItem的数据的类。
public class TestListViewItemClass
{
public int TestInt { get; set; }
public string TestString { get; set; }
public DateTime TestDateTime { get; set; }
public TimeSpan TestTimeSpan { get; set; }
public decimal TestDecimal { get; set; }
}
然后,创建一个返回数据项的方法。此方法可以查询数据库,调用Web服务API或其他任何方法,只要它返回类类型的IEnumerable即可。
public IEnumerable<TestListViewItemClass> GetItems()
{
for (int x = 0; x < 15000; x++)
{
yield return new TestListViewItemClass()
{
TestDateTime = DateTime.Now,
TestTimeSpan = TimeSpan.FromDays(x),
TestInt = new Random(DateTime.Now.Millisecond).Next(),
TestDecimal = (decimal)x + new Random(DateTime.Now.Millisecond).Next(),
TestString = "Test string " + x,
};
}
}
最后,在ListView所在的窗体上,您可以填充ListView。为了演示目的,我使用表单的Load事件来填充ListView。更可能的是,您会希望在表单上的其他位置执行此操作。
我已经包含了从我的类的一个实例TestListViewItemClass生成ListViewItem的函数。在生产场景中,您可能需要在其他地方定义该功能。
private async void TestListViewForm_Load(object sender, EventArgs e)
{
var function = new Func<TestListViewItemClass, ListViewItem>((TestListViewItemClass x) =>
{
var item = new ListViewItem();
if (x != null)
{
item.Text = x.TestString;
item.SubItems.Add(x.TestDecimal.ToString("F4"));
item.SubItems.Add(x.TestDateTime.ToString("G"));
item.SubItems.Add(x.TestTimeSpan.ToString());
item.SubItems.Add(x.TestInt.ToString());
item.Tag = x;
return item;
}
return null;
});
PopulateListView<TestListViewItemClass>(this.listView1, function, GetItems(), progress);
}
在上面的例子中,我创建的窗体的构造函数的IProgress对象是这样的:
progress = new Progress<int>(value =>
{
toolStripProgressBar1.Visible = true;
if (value >= 100)
{
toolStripProgressBar1.Visible = false;
toolStripProgressBar1.Value = 0;
}
else if (value > 0)
{
toolStripProgressBar1.Value = value;
}
});
我使用的项目填充一个ListView很多次,我们被填充起来的这种方法到ListView中的12,000个项目,并且速度非常快。主要的是你需要从数据库完全构建你的对象,然后你甚至可以触摸ListView进行更新。
希望这是有帮助的。
我在下面包含了该方法的异步版本,该版本调用了此帖子顶部显示的主要方法。
public static Task PopulateListViewAsync<T>(ListView listView, Func<T, ListViewItem> func,
IEnumerable<T> objects, IProgress<int> progress) where T : class, new()
{
return Task.Run(() => PopulateListView<T>(listView, func, objects, progress));
}
冻结意味着别的东西:它意味着对象(在这种情况下是元素的集合)在冻结时不会改变。在这种情况下,你马上修改它! – 2010-07-21 12:16:07
冻结只是一个术语,用于解释我的要求 – 2010-07-21 12:26:22