WinForm ListView 大数据提高加载速度的方法 虚拟模式加载

作者:袖梨 2022-06-25

将VirtualMode 属性设置为 true 会将 ListView 置于虚拟模式。控件不再使用Collection.Add()这种方式来添加数据,取而代之的是使用RetrieveVirtualItem(Occurs when the ListView is in virtual mode and requires a ListViewItem.)和CacheVirtualItems两个事件,单独使用RetrieveVirtualItem也可以,CacheVirtualItems这个事件主要是为了方便编程人员操作缓冲集合,其参数CacheVirtualItemsEventArgs有StartIndex和EndIndex两个属性在虚拟模式下。

在虚拟模式下,从缓冲之中获取所需的数据进行加载,性能会有很大提高。 在其他情况下,可能需要经常重新计算 ListViewItem 对象的值,对整个集合进行此操作将产生不可接受的性能。

示例代码:

using System;
using System.Collections.Generic;
using System.Windows.Forms;
namespace WinFormTest
{
    public partial class Form1 : Form
    {
        private List myCache;
        public Form1()
        {
            InitializeComponent();
            myCache = new List();
        }
        private void Form1_Load(object sender, EventArgs e)
        {
            listView1.View = View.Details;
            listView1.VirtualMode = true;
            listView1.RetrieveVirtualItem += new RetrieveVirtualItemEventHandler(listView1_RetrieveVirtualItem);
        }
        void listView1_RetrieveVirtualItem(object sender, RetrieveVirtualItemEventArgs e)
        {
            if (myCache != null )
            {
                e.Item = myCache[e.ItemIndex];
            }
            else
            {
                //A cache miss, so create a new ListViewItem and pass it back.
                int x = e.ItemIndex * e.ItemIndex;
                e.Item = new ListViewItem(x.ToString());
            }
        }
        
        private void button1_Click(object sender, EventArgs e)
        {
            List list = GetStudentList();
            foreach (var item in list)
            {
                ListViewItem listViewItem = new ListViewItem();
                listViewItem.SubItems[0].Text = item.Name;
                listViewItem.SubItems.Add(item.Sex);
                myCache.Add(listViewItem);
            }
            listView1.VirtualListSize = myCache.Count;
        }
        private List GetStudentList()
        {
            List list = new List();
            for (int i = 0; i 


总结

  (1)必须设置VirtualMode为true并设置VirtualListSize大小

  (2)绑定该事件RetrieveVirtualItem

  (3)如果中间更新了数据需要重新设置VirtualListSize,并调用Invalidate()方法

  (4)禁用selectedItem,在该模式下使用selectedItem将产生异常,可以用下面方法代替

private List FindSelectedAll()
{
    List r = new List();
    foreach (int item in listView1.SelectedIndices)
    {
         r.Add(bufferItems[item]);
     }
     return r;
 }





WinForm ListView不分页加载大量数据

WinForm的ListView在加载大量数据时会出现闪烁的问题,同时数据加载很慢。如果你的列表中有超过千条的数据且不做特殊处理还是用普通的ListView.Items.Add(),估计你的用户得抱怨了。

下面说下解决方法:

1、使用listView1.Items.AddRange()代替Add

这种方法需要先将数据放入一个缓存数组中,然后调用AddRange一次性加入ListView中,同时可以用一个计数器记录一次性加入缓存的数量,如下:


listView1.Items.Clear();
if (vList.Count > 0)
{

int indexI = 0;
List listBuffer = new List();
foreach (var item in vList)
{
ListViewItem li = new ListViewItem();
li.ImageIndex = 0;
li.SubItems[0].Text = item.Name;
li.Tag = item;
li.ForeColor = item.Status == 0 ? Color.Green : Color.Red;
listBuffer.Add(li);

if (indexI++ % 1000 == 0)
{
listView1.Items.AddRange(listBuffer.ToArray());
listBuffer.Clear();
}
if (indexI % 50 == 0)
{
Application.DoEvents();
}
}
listView1.Items.AddRange(listBuffer.ToArray());

}

这样可以减少ListView闪烁的次数,数据量不是很大时有效果。


2、自定义ListView类

下面这个类在网上广为流传,虽然解决了ListView闪烁的问题,但是在打开速度上和原来没什么区别,同时带来一个问题就是如果程序切换到别的ListView上,数据还会继续忘原来的ListView中添加,直到数据全面添加完成。


public class ListViewLargeData : System.Windows.Forms.ListView
{
public ListViewLargeData()
{
this.SetStyle(ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint, true);
this.SetStyle(ControlStyles.EnableNotifyMessage, true);
}
protected override void OnNotifyMessage(Message m)
{
if (m.Msg != 0x14)
{
base.OnNotifyMessage(m);
}
}
}

3、开启ListView的VirtualMode模式


此方式也是本文重点推荐的方式,可以实现不闪烁效果,而且打开速度很快。往往好的效果带来的问题是代码较为复杂,具体写法可以看官方示例http://msdn.mic**ros*oft.com/zh-cn/library/system.windows.forms.listview.virtualmode.aspx,下面说下需要注意的几点:

(1)必须设置VirtualMode为true并设置VirtualListSize大小

listView1.VirtualMode = true;
listView1.VirtualListSize = bufferItems.Count;
listView1.RetrieveVirtualItem += new RetrieveVirtualItemEventHandler(listView_RetrieveVirtualItem);

(2)绑定该事件为ListView计算Item

void listView_RetrieveVirtualItem(object sender, RetrieveVirtualItemEventArgs e)
{
e.Item = m_hListViewItems[e.ItemIndex];
}

(3)如果中间更新了数据需要重新设置VirtualListSize,并调用Invalidate()方法(此方法并非必须请高手指点)。

listView1.VirtualListSize = bufferItems.Count;
listView1.Invalidate();

(4)禁用selectedItem,在该模式下使用selectedItem将产生异常,可以用下面方法代替

private List FindSelectedAll()
{
List r = new List();
foreach (int item in listView1.SelectedIndices)
{
r.Add(bufferItems[item]);
}
return r;
}

相关文章

精彩推荐