CListView虚拟列表

首先说下虚拟列表出现的原因:

数据量比较小的时候,对于CListView控件可以直接使用InsertItem并配合SetItemText函数来插入并修改数据。这样操作很直接。

但是,如果数据量比较大了,比如1w个数据,那么,根据插入的数据种类,长度,以及计算机性能,估计时间在10s中到1分钟不等。如果你的用户在使用的这样的程序时肯定不会开心,初始化的时候插入则需要登上10s+后才能使用,如果外加一个线程来并行插入,倒也是一个方法(这个事情我做过),不过,很浪费CPU时间,以及内存。

如果是10w个,甚至100w个数据呢?那么至少会线性的增加时间了~

但是回头想一想,一个列表,在你的计算机显示器上,最多也就能看到50-80个。我的笔记本分辨率是1366*768的就按50个来算,768/50 约等于15吧,一行用15个像素宽度表示,已经有点小勉强了~~~

所以这个时候有个事情就非常明显了:在大数据量的时候,根本没必要在初始化时把全部的数据都插入到CListCtrl控件中。

要注意,这个问题是我在下自习回寝室的路上想明白的~~~

然后我就想了,既然这样的话,如果自己动手做虚拟列表,也不是不能做。基本上需要准备好以下的东西:

1 右面的浏览滑块,就是那个scroll,这个需要处理好,根据不同的位置,动态加载不同地方的数据。

2 鼠标的滚轮消息,上下移动,也要做好。

3 窗口的最大数据量,以及文字显示之类的东西,都要做。

基本上就是意味着需要自己做一个控件,虚拟列表控件。这个确实是可行的。然后呢~我在百度查资料的时候,意外的看到了虚拟列表这个东西~~~

那时我才知道,原来microsoft的大神们已经想到了这个问题,而且在CListCtrl中已经整合好了。

在CListCtrl中使用虚拟列表

我之前的那篇关于CListCtrl控件使用方法的文章中说过,对于和控件绑定了的CListCtrl对象,主要需要做的工作,就是设置风格,并且插入列。

不过对于虚拟列表,与常规的列表相比,并没有什么风格上的不同,所以风格还是依照自己的需求进行设置就可以了。

列的控制,按照前面那篇文章上来就好了~这里不是重点。

然后呢,很需要做的一点就是,设置最大条目的数量。这里要注意的是,所谓的最大条目的数量,就是和你的数据库的数据量。这两者一定是要匹配的。

展示一小段实例代码:

 1 else
 2     {
 3         CFileInfo cfi;
 4         while(m_MyDataBase.ReadString(path))
 5         {
 6             cfi.csFileRoot = path;
 7             cfi.csFileName = path.Right(path.GetLength() - path.ReverseFind('\\') - 1);
 8             cfi.csFilePath = path.Left(path.ReverseFind('\\'));
 9             m_arrayFileInfo.Add(cfi);
10             nFileNum++;
11         }
12         m_MyDataBase.Close();
13     }
14     
15     m_LCTable.SetItemCountEx(nFileNum, LVSICF_NOSCROLL|LVSICF_NOINVALIDATEALL);
16     nInitialFlag = 1;

最上面的代码是数据初始化的部分,存储到CArray模板中。也算是个最简单的数据库吧。

初始化过后,我的数据就放到了容器中,上面的nFileNum变量就是跟踪容量的数据,不过,其实也不需要,毕竟CArray是提供数据总量查询的。

然后接下来调用我们的明星函数:SetItemCountEx

这个函数,第一个参数毫无疑问就是设置数量上限的,第二个参数又是一个什么风格设置,看看MSDN怎么说:

    • LVSICF_NOINVALIDATEALL The list view control will not repaint unless affected items are currently in view. This is the default value.

    • LVSICF_NOSCROLL The list view control will not change the scroll position when the item count changes. 

大意我就不翻译了,现在有个人正在和我聊天,翻译了太费时间。这两个的特性可以自己试一试。

然后在这算是第一步完成,主要还是SetItemCountEx函数。

 

接下来,需要做的事进行消息响应。

要知道,windows程序的运行是需要消息来推动的。当滑块对拖动或者鼠标滚动的时候,都会有消息产生。

所以,CListCtrl也是采用了这种机制,消息响应的方式去填充虚拟列表。

我在这里只介绍我使用过的一个消息,就是LVN_GETDISPINFO

最上面的那个消息响应就是了。

然后我们看响应函数的代码:

 1 void CMyRisingDlg::OnLvnGetdispinfoList2(NMHDR *pNMHDR, LRESULT *pResult)
 2 {
 3     NMLVDISPINFO *pDispInfo = reinterpret_cast<NMLVDISPINFO*>(pNMHDR);
 4     // TODO: Add your control notification handler code here
 5     LV_DISPINFO * pLPD = (LV_DISPINFO *)pDispInfo;
 6     LV_ITEM* pItem= &(pDispInfo)->item;
 7     HICON hIconTmp;
 8     int nItem = pItem->iItem;
 9     if (pItem->mask & LVIF_TEXT) //valid text buffer?
10     {
11        switch(pItem->iSubItem)
12        {
13           case 0: //fill in main text首列添加图像的工作肯定也要在这里完成
14              _tcscpy(pItem->pszText, m_arrayFileInfo[nItem].csFileName);
15              //ExtractIconEx(m_arrayFileInfo[nItem].csFileRoot, 0, NULL, &hIconTmp, 1);
16              //m_imagelist.Add(hIconTmp);
17              //Add Icon into the list
18              pItem->iImage = 0;
19              //m_imagelist.Remove(0);
20              break;
21           case 1: //fill in sub item 1 text
22              _tcscpy(pItem->pszText, m_arrayFileInfo[nItem].csFilePath);
23              break;
24           case 2: //fill in sub item 2 text
25              _tcscpy(pItem->pszText, m_arrayFileInfo[nItem].FileTime);
26              break;
27        }
28     }
29 
30     *pResult = 0;
31 }

首先能看参数,其中一个是NMHDR的指针,这个类型,我还真的不是很清楚。

不过清楚的可以看到,经过两次类型转换,我们会得到一个LV_ITEM的指针。在这简单的说一下,前面的那段类型转换的代码,是函数自动生成的时候就已经改出来的。

接着,在这里我们只要根据LV_ITEM中的消息进行相应来添加相应的数据就可以了。

所以接下来主要需要看的就是switch中的case选择。

在上面代码中的switchcase结构中,传给switch的参数就是一个条目中的子序号。根据这个子序号,去添加对应位置的信息。

然后这里还有非常重要的一点,就是在上面的这段代码:

 8     int nItem = pItem->iItem;

这段代码就是用来获取当前位置对应数据的目录号。只有根据这个目录号,你才能找到在你的数据中应该添加的数据的位置。这个在我的代码中你应该也能注意到的。

最主要的两个数据在这里基本就介绍完了,具体的代码也贴在上面。这样就能进行最简单的虚拟列表操作啦。

 

posted @ 2013-07-12 11:23  Matrix_R  阅读(3185)  评论(1编辑  收藏  举报