在實現繪圖的過程中,顯示的圖形總是會閃爍,筆者曾經被這個問題折磨了好久,通過向高手請教,搜索資料,問題基本解決,現將文檔整理出來以供大家參考.
1.顯示的圖形為什麼會閃爍
我們的繪圖過程大多放在OnDraw或者OnPaint函數中,OnDraw在進行屏幕顯示時是由OnPaint進行調用的。當窗口由於任何原因需要重繪時,總是先用背景色將顯示區清除,然後才調用OnPaint,而背景色往往與繪圖內容反差很大,這樣在短時間內背景色與顯示圖形的交替出現,使得顯示窗口看起來在閃。如果將背景刷設置成NULL,這樣無論怎樣重繪圖形都不會閃了。當然,這樣做會使得窗口的顯示亂成一團,因為重繪時沒有背景色對原來繪制的圖形進行清除,而又疊加上了新的圖形。有的人會說,閃爍是因為繪圖的速度太慢或者顯示的圖形太復雜造成的,其實這樣說並不對,繪圖的顯示速度對閃爍的影響不是根本性的。例如在OnDraw(CDC
*pDC)中這樣寫:
pDC->MoveTo(0,0);
pDC->LineTo(100,100);
for(int i=0;i<100000;i++)
{
pDC->MoveTo(0,i);
pDC->LineTo(1000,i);
}
程序有點極端,但是能說明問題。
說到這裡可能又有人要說了,為什麼一個簡單圖形看起來沒有復雜圖形那麼閃呢?這是因為復雜圖形占的面積大,重畫時造成的反差比較大,所以感覺上要閃得厲害一些,但是閃爍頻率要低。那為什麼動畫的重畫頻率高,而看起來卻不閃?這裡,我就要再次強調了,閃爍是什麼?閃爍就是反差,反差越大,閃爍越厲害。因為動畫的連續兩個幀之間的差異很小所以看起來不閃。如果不信,可以在動畫的每一幀中間加一張純白的幀,不閃才怪呢。
2、如何避免閃爍
在知道圖形顯示閃爍的原因之後,對症下藥就好辦了。首先當然是去掉MFC提供的背景繪制過程了。實現的方法很多:
* 可以在窗口形成時給窗口的注冊類的背景刷付NULL
* 也可以在形成以後修改背景
static CBrush brush(RGB(255,0,0));
SetClassLong(this->m_hWnd,GCL_HBRBACKGROUND,(LONG)(HBRUSH)brush);
* 要簡單也可以重載OnEraseBkgnd(CDC* pDC)直接返回TRUE
這樣背景沒有了,結果圖形顯示的確不閃了,但是顯示也象前面所說的一樣,變得一團亂。怎麼辦?這就要用到雙緩存的方法了。雙緩沖就是除了在屏幕上有圖形進行顯示以外,在內存中也有圖形在繪制。我們可以把要顯示的圖形先在內存中繪制好,然後再一次性的將內存中的圖形按照一個點一個點地覆蓋到屏幕上去(這個過程非常快,因為是非常規整的內存拷貝)。這樣在內存中繪圖時,隨便用什麼反差大的背景色進行清除都不會閃,因為看不見。當貼到屏幕上時,因為內存中最終的圖形與屏幕顯示圖形差別很小(如果沒有運動,當然就沒有差別),這樣看起來就不會閃。
3、如何實現雙緩沖
首先給出實現的程序,然後再解釋,同樣是在OnDraw(CDC *pDC)中:
CDC MemDC; //首先定義一個顯示設備對象
CBitmap MemBitmap;//定義一個位圖對象
//隨後建立與屏幕顯示兼容的內存顯示設備
MemDC.CreateCompatibleDC(NULL);
//這時還不能繪圖,因為沒有地方畫
//下面建立一個與屏幕顯示兼容的位圖,至於位圖的大小嘛,可以用窗口的大小
MemBitmap.CreateCompatibleBitmap(pDC,nWidth,nHeight);
//將位圖選入到內存顯示設備中
//只有選入了位圖的內存顯示設備才有地方繪圖,畫到指定的位圖上
CBitmap *pOldBit=MemDC.SelectObject(&MemBitmap);
//先用背景色將位圖清除干淨,這裡我用的是白色作為背景
//你也可以用自己應該用的顏色
MemDC.FillSolidRect(0,0,nWidth,nHeight,RGB(255,255,255));
//繪圖
MemDC.MoveTo(……);
MemDC.LineTo(……);
//將內存中的圖拷貝到屏幕上進行顯示
pDC->BitBlt(0,0,nWidth,nHeight,&MemDC,0,0,SRCCOPY);
//繪圖完成後的清理
MemBitmap.DeleteObject();
MemDC.DeleteDC();
///////////////
無閃爍背景圖繪制 //////////
//// 程序設計: icemen (溫冰) 樹愛兵
//////////////////////////////////////////////
BOOL CStrucView::OnEraseBkgnd(CDC* pDC)
{ int nWidth;
int nHeight;
//CView::OnEraseBkgnd(pDC);
CStrucDoc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
CRect rect;
GetWindowRect(&rect);
nWidth = rect.Width();
nHeight= rect.Height();
CDC MemDC;
CBitmap MemBitmap;
MemDC.CreateCompatibleDC (NULL);
MemBitmap.CreateCompatibleBitmap(pDC,nWidth,nHeight);
CBitmap *pOldBit=MemDC.SelectObject(&MemBitmap);
MemDC.FillSolidRect(0,0,nWidth,nHeight,RGB(255,255,255));
///////////////////////////////////////////////////////
//// 以上為畫背景色 //////////////
//// 以下為畫背景圖 //////////////
///////////////////////////////////////////////////////
GetClientRect(rect);
BITMAP bm;
CDC dcMem;
VERIFY(m_bmp.GetObject(sizeof(bm),(LPVOID)&bm));
dcMem.CreateCompatibleDC(pDC);
CBitmap *pOldBMP =( CBitmap *)dcMem.SelectObject(&m_bmp);
MemDC.BitBlt( (rect.right - bm.bmWidth)/2,
(rect.bottom - bm.bmHeight)/2,
bm.bmWidth,bm.bmHeight,&dcMem,0,0,SRCCOPY);
dcMem.SelectObject(pOldBMP);
///////////////////////////////////////////////////////
//// 以上為畫背景圖 //////////////
///////////////////////////////////////////////////////
pDC->BitBlt(0,0,nWidth,nHeight,&MemDC,0,0,SRCCOPY);
MemBitmap.DeleteObject();
MemDC.DeleteDC();
return TRUE;
}
/////////////// 無閃爍背景圖繪制 //////////
/////////////////////////////////////////////
protected:
CBitmap m_bmp;
CStrucView::CStrucView()
{
VERIFY(m_bmp.LoadBitmap(IDB_BITMAP3));
}