VC++ MFC CListCtrl 插入大量数据 使用 虚拟 CListCtrl的问题。
虽然虚拟CListCtrl解决了插入大量数据非常慢的问题,但是在虚拟CListCtrl中选中多页数据的时候也会变得很慢,每次选中多个数据翻页的时候都会很慢,能不能解决选中...
虽然虚拟 CListCtrl解决了插入大量数据 非常慢的问题,但是在虚拟CListCtrl中选中多页数据的时候也会变得很慢,每次选中多个数据翻页的时候都会很慢,能不能解决选中多个数据慢或者选择其他方法解决 插入大量数据慢并且选中快的。
展开
2个回答
展开全部
1.使用SetRedraw禁止窗口重绘,操作完成后,再恢复窗口重绘
m_ctlList.SetRedraw(FALSE);
//以下为更新数据操作
//……
//恢复窗口重绘
m_ctlList.SetRedraw(TRUE);
2.使用LockWindowUpdate禁止窗口重绘,操作完成后,用UnlockWindowUpdate恢复窗口重绘
m_ctlList.LockWindowUpdate();
//以下为更新数据操作
//……
//恢复窗口重绘
m_ctlList.UnlockWindowUpdate();
3.使用ListCtrl的内部双缓冲
m_ctlLisit.SetExtendedStyle(m_ctlLisit.GetExtendedStyle()|LVS_EX_DOUBLEBUFFER);
VC6未定义LVS_EX_DOUBLEBUFFER宏,使用者可以自定义,如下:
#define LVS_EX_DOUBLEBUFFER 0x00010000
4.Virtual List
首先要设置ListCtrl风格为LVS_REPORT | LVS_OWNERDATA或在ListCtrl属里中的More Styles页面中选中Owner data复选框。
其次要向应LVN_GETDISPINFO消息;
void OnGetdispinfoList(NMHDR* pNMHDR, LRESULT* pResult)
{
LV_DISPINFO* pDispInfo = (LV_DISPINFO*)pNMHDR;
LV_ITEM *pItem = &(pDispInfo)->item;
char szText[128] = {0};
if (pItem->mask & LVIF_TEXT)
{
//使缓冲区数据与表格子项对应
//m_ArrayBuff为二维数组
//定义如下 int m_ArrayBuff[2048][4];
_stprintf(szText,_T("%d"),m_ArrayBuff[pItem->iItem][pItem->iSubItem]);
pItem->pszText = szText;
}
*pResult = 0;
}
最后便是生成缓冲区数据
void Insertdata()
{
//删除之前的数据
m_ctlList.SetItemCountEx(0);
m_ctlList.Invalidate();
m_ctlList.UpdateWindow();
srand( (unsigned)time( NULL ));
//生成新的数据缓冲区
int nItemCount = 2048;
for (int i = 0;i < nItemCount; i ++)
{
for (int k = 0;k < 4;k ++)
{
m_ArrayBuff[i][k] = rand()%2048 + 1;
}
}
if (nItemCount < 2)
m_ctlList.SetItemCountEx(1);
else
m_ctlList.SetItemCountEx(nItemCount);
m_ctlList.Invalidate();
}
若要修改数据,只要修改缓冲区m_ArrayBuff的数据即可以
5.Custom Redraw
既然是自绘,首先当然是重载CListCtrl类,并接管WM_ERASEBKGND消息,去掉默认的处理,改为不处理
BOOL CListCtrlEx::OnEraseBkgnd(CDC* pDC)
{
//响应WM_ERASEBKGND消息
return false;
//屏蔽默认处理
//return CListCtrl::OnEraseBkgnd(pDC);
}
void CListCtrlEx::OnPaint()
{
//响应WM_PAINT消息
CPaintDC dc(this); // device context for painting
CRect rect;
CRect headerRect;
CDC MenDC;//内存ID表
CBitmap MemMap;
GetClientRect(&rect);
GetDlgItem(0)->GetWindowRect(&headerRect);
MenDC.CreateCompatibleDC(&dc);
MemMap.CreateCompatibleBitmap(&dc,rect.Width(),rect.Height());
MenDC.SelectObject(&MemMap);
MenDC.FillSolidRect(&rect,RGB(228,236,243));
//这一句是调用默认的OnPaint(),把图形画在内存DC表上
DefWindowProc(WM_PAINT,(WPARAM)MenDC.m_hDC,(LPARAM)0);
//输出
dc.BitBlt(0,headerRect.Height(),rect.Width(), rect.Height(),&MenDC,0, headerRect.Height(),SRCCOPY);
MenDC.DeleteDC();
MemMap.DeleteObject();
}
m_ctlList.SetRedraw(FALSE);
//以下为更新数据操作
//……
//恢复窗口重绘
m_ctlList.SetRedraw(TRUE);
2.使用LockWindowUpdate禁止窗口重绘,操作完成后,用UnlockWindowUpdate恢复窗口重绘
m_ctlList.LockWindowUpdate();
//以下为更新数据操作
//……
//恢复窗口重绘
m_ctlList.UnlockWindowUpdate();
3.使用ListCtrl的内部双缓冲
m_ctlLisit.SetExtendedStyle(m_ctlLisit.GetExtendedStyle()|LVS_EX_DOUBLEBUFFER);
VC6未定义LVS_EX_DOUBLEBUFFER宏,使用者可以自定义,如下:
#define LVS_EX_DOUBLEBUFFER 0x00010000
4.Virtual List
首先要设置ListCtrl风格为LVS_REPORT | LVS_OWNERDATA或在ListCtrl属里中的More Styles页面中选中Owner data复选框。
其次要向应LVN_GETDISPINFO消息;
void OnGetdispinfoList(NMHDR* pNMHDR, LRESULT* pResult)
{
LV_DISPINFO* pDispInfo = (LV_DISPINFO*)pNMHDR;
LV_ITEM *pItem = &(pDispInfo)->item;
char szText[128] = {0};
if (pItem->mask & LVIF_TEXT)
{
//使缓冲区数据与表格子项对应
//m_ArrayBuff为二维数组
//定义如下 int m_ArrayBuff[2048][4];
_stprintf(szText,_T("%d"),m_ArrayBuff[pItem->iItem][pItem->iSubItem]);
pItem->pszText = szText;
}
*pResult = 0;
}
最后便是生成缓冲区数据
void Insertdata()
{
//删除之前的数据
m_ctlList.SetItemCountEx(0);
m_ctlList.Invalidate();
m_ctlList.UpdateWindow();
srand( (unsigned)time( NULL ));
//生成新的数据缓冲区
int nItemCount = 2048;
for (int i = 0;i < nItemCount; i ++)
{
for (int k = 0;k < 4;k ++)
{
m_ArrayBuff[i][k] = rand()%2048 + 1;
}
}
if (nItemCount < 2)
m_ctlList.SetItemCountEx(1);
else
m_ctlList.SetItemCountEx(nItemCount);
m_ctlList.Invalidate();
}
若要修改数据,只要修改缓冲区m_ArrayBuff的数据即可以
5.Custom Redraw
既然是自绘,首先当然是重载CListCtrl类,并接管WM_ERASEBKGND消息,去掉默认的处理,改为不处理
BOOL CListCtrlEx::OnEraseBkgnd(CDC* pDC)
{
//响应WM_ERASEBKGND消息
return false;
//屏蔽默认处理
//return CListCtrl::OnEraseBkgnd(pDC);
}
void CListCtrlEx::OnPaint()
{
//响应WM_PAINT消息
CPaintDC dc(this); // device context for painting
CRect rect;
CRect headerRect;
CDC MenDC;//内存ID表
CBitmap MemMap;
GetClientRect(&rect);
GetDlgItem(0)->GetWindowRect(&headerRect);
MenDC.CreateCompatibleDC(&dc);
MemMap.CreateCompatibleBitmap(&dc,rect.Width(),rect.Height());
MenDC.SelectObject(&MemMap);
MenDC.FillSolidRect(&rect,RGB(228,236,243));
//这一句是调用默认的OnPaint(),把图形画在内存DC表上
DefWindowProc(WM_PAINT,(WPARAM)MenDC.m_hDC,(LPARAM)0);
//输出
dc.BitBlt(0,headerRect.Height(),rect.Width(), rect.Height(),&MenDC,0, headerRect.Height(),SRCCOPY);
MenDC.DeleteDC();
MemMap.DeleteObject();
}
展开全部
要自己写的扩展类
主要部分的代码,我给你一个清单
/ / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
/ /添加背景色
无效CXXSortListCtrl :: OnCustomDraw(NMHDR * pNMHDR,LRESULT * pResult)
{
LPNMLVCUSTOMDRAW lplvcd =(LPNMLVCUSTOMDRAW)的pNMHDR;
开关(lplvcd - > nmcd.dwDrawStage)
{
情况下CDDS_PREPAINT:
* pResult = CDRF_NOTIFYSUBITEMDRAW / /问子项通知。
突破;
情况下CDDS_ITEMPREPAINT:/ /子项通知要求。
* pResult = CDRF_NOTIFYSUBITEMDRAW;
(GetExtendedStyle()LVS_EX_CHECKBOXES)/ /如果我们有一个复选框风格,
{/ /忘记子项目通知。
* pResult = CDRF_DODEFAULT
诠释iRow = lplvcd - > nmcd.dwItemSpec;
((iRow 1))的/ / &&(ICOL 1))/ /单线
{
lplvcd:> clrTextBk = m_clrLine / / RGB(255,0,0);
lplvcd->的clrText m_clrText / / RGB(255,255,0); BR /> * pResult = CDRF_NEWFONT;
}
在else / /双列
{
lplvcd - > clrTextBk = m_clrTwoLine; / / RGB(255,0,0);
lplvcd - > clrText = m_clrText; / / RGB(255,255,0);
* pResult = CDRF_NEWFONT;
}
/ /注意... ...... :)
/ /这里是你想改变的地方...
(GetCheck(iRow)),/ /高光检查行
{
/ / lplvcd - > clrTextBk m_clrChkText; / / RGB(255,0,0);
lplvcd - > clrText = m_clrChkText; / / RGB(255,255,0);
* pResult = CDRF_NEWFONT;
}
}
突破;
情况下CDDS_ITEMPREPAINT:| CDDS_SUBITEM:/ / RECD时CDRF_NOTIFYSUBITEMDRAW中返回
{/ /响应CDDS_ITEMPREPAINT。
* pResult = CDRF_DODEFAULT
诠释ICOL的= lplvcd - > iSubItem;
诠释iRow = lplvcd - > nmcd.dwItemSpec;
((iRow 1))的/ / && (ICOL 1))/ /单线路
{
lplvcd clrTextBk:= m_clrLine / / RGB(255,0,0);
lplvcd> clrText = m_clrText; / / RGB (255,255,0);
* pResult = CDRF_NEWFONT;
}
其他/ /双列
{
lplvcd clrTextBk = m_clrTwoLine / / RGB (255,0,0);
lplvcd - > clrText = m_clrText; / / RGB(255,255,0);
* pResult = CDRF_NEWFONT;
}
(GetItemState(iRow,LVIS_SELECTED))
{
lplvcd> clrTextBk = m_clrSelected / / RGB(255,0,0);
lplvcd - > clrText = m_clrSelText / / RGB( 255,255,0);
* pResult = CDRF_NEWFONT;
}
突破;
}
默认:/ /这不是一个通知这是有趣的。
* pResult = CDRF_DODEFAULT;}
}
主要部分的代码,我给你一个清单
/ / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / / /
/ /添加背景色
无效CXXSortListCtrl :: OnCustomDraw(NMHDR * pNMHDR,LRESULT * pResult)
{
LPNMLVCUSTOMDRAW lplvcd =(LPNMLVCUSTOMDRAW)的pNMHDR;
开关(lplvcd - > nmcd.dwDrawStage)
{
情况下CDDS_PREPAINT:
* pResult = CDRF_NOTIFYSUBITEMDRAW / /问子项通知。
突破;
情况下CDDS_ITEMPREPAINT:/ /子项通知要求。
* pResult = CDRF_NOTIFYSUBITEMDRAW;
(GetExtendedStyle()LVS_EX_CHECKBOXES)/ /如果我们有一个复选框风格,
{/ /忘记子项目通知。
* pResult = CDRF_DODEFAULT
诠释iRow = lplvcd - > nmcd.dwItemSpec;
((iRow 1))的/ / &&(ICOL 1))/ /单线
{
lplvcd:> clrTextBk = m_clrLine / / RGB(255,0,0);
lplvcd->的clrText m_clrText / / RGB(255,255,0); BR /> * pResult = CDRF_NEWFONT;
}
在else / /双列
{
lplvcd - > clrTextBk = m_clrTwoLine; / / RGB(255,0,0);
lplvcd - > clrText = m_clrText; / / RGB(255,255,0);
* pResult = CDRF_NEWFONT;
}
/ /注意... ...... :)
/ /这里是你想改变的地方...
(GetCheck(iRow)),/ /高光检查行
{
/ / lplvcd - > clrTextBk m_clrChkText; / / RGB(255,0,0);
lplvcd - > clrText = m_clrChkText; / / RGB(255,255,0);
* pResult = CDRF_NEWFONT;
}
}
突破;
情况下CDDS_ITEMPREPAINT:| CDDS_SUBITEM:/ / RECD时CDRF_NOTIFYSUBITEMDRAW中返回
{/ /响应CDDS_ITEMPREPAINT。
* pResult = CDRF_DODEFAULT
诠释ICOL的= lplvcd - > iSubItem;
诠释iRow = lplvcd - > nmcd.dwItemSpec;
((iRow 1))的/ / && (ICOL 1))/ /单线路
{
lplvcd clrTextBk:= m_clrLine / / RGB(255,0,0);
lplvcd> clrText = m_clrText; / / RGB (255,255,0);
* pResult = CDRF_NEWFONT;
}
其他/ /双列
{
lplvcd clrTextBk = m_clrTwoLine / / RGB (255,0,0);
lplvcd - > clrText = m_clrText; / / RGB(255,255,0);
* pResult = CDRF_NEWFONT;
}
(GetItemState(iRow,LVIS_SELECTED))
{
lplvcd> clrTextBk = m_clrSelected / / RGB(255,0,0);
lplvcd - > clrText = m_clrSelText / / RGB( 255,255,0);
* pResult = CDRF_NEWFONT;
}
突破;
}
默认:/ /这不是一个通知这是有趣的。
* pResult = CDRF_DODEFAULT;}
}
本回答被提问者采纳
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询