MFC中螢幕重新整理閃爍問題解決方法總結

2021-06-20 05:02:31 字數 3645 閱讀 9313

在mfc中經常遇到螢幕重新整理的問題,下面是幾種解決方法。

1,重新整理整個程式區域,有明顯的閃爍情況

invalidate( true );

updatewindow(); 

2,重新整理指定區域,該區域有閃爍情況,不過比上一種方法要好多了。

invalidaterect( &m_rect, true 0);

updatewindow(); 

3,還有一種方法

invalidaterect( &m_rect, true 0);

redrawwindow( &m_rect, null, rdw_internalpaint |

rdw_invalidate | rdw_updatenow

| rdw_erase );

4, 最好的一種方法是使用 bitblt 函式來拷貝需要重新整理的區域。

5,最後,當然也可以直接使用 onpaint() 放置在要重新整理的位置。

無閃爍刷屏技術的實現

1.    顯示的圖形為什麼會閃爍?

我們的繪圖過程大多放在ondraw或者onpaint函式中,ondraw在進行螢幕顯示時是由onpaint進行呼叫的。

當視窗由於任何原因需要重繪時,總是先用背景色將顯示區清除,然後才呼叫onpaint,而背景色往往與繪圖內容反差很大,這樣在短時間內背景色與顯示圖形的交替出現,使得顯示視窗看起來在閃。如果將背景刷設定成null,這樣無論怎樣重繪圖形都不會閃了。當然,這樣做會使得視窗的顯示亂成一團,因為重繪時沒有背景色對原來繪製的圖形進行清除,而又疊加上了新的圖形。有的人會說,閃爍是因為繪圖的速度太慢或者顯示的圖形太複雜造成的,其實這樣說並不對,繪圖的顯示速度對閃爍的影響不是根本性的。例如在ondraw(cdc *pdc)中這樣寫:

pdc->moveto(0,0);

pdc->lineto(100,100);

這個繪圖過程應該是非常簡單、非常快了吧,但是拉動視窗變化時還是會看見閃爍。其實從道理上講,畫圖的過程越複雜越慢閃爍應該越少,因為繪圖用的時間與用背景清除螢幕所花的時間的比例越大人對閃爍的感覺會越不明顯。比如:清楚螢幕時間為1s繪圖時間也是為1s,這樣在10s內的連續重畫中就要閃爍5次;如果清楚螢幕時間為1s不變,而繪圖時間為9s,這樣10s內的連續重畫只會閃爍一次。這個也可以試驗,在ondraw(cdc *pdc)中這樣寫:

for(int i=0;i<100000;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();

上面的注釋應該很詳盡了,廢話就不多說了。

4、如何提高繪圖的效率

實際上,在ondraw(cdc *pdc)中繪製的圖並不是所有都顯示了的,例如:你在ondraw中畫了兩個矩形,在一次重繪中雖然兩個矩形的繪製函式都有執行,但是很有可能只有乙個顯示了,這是因為mfc本身為了提高重繪的效率設定了裁剪區。裁剪區的作用就是:只有在這個區內的繪圖過程才會真正有效,在區外的是無效的,即使在區外執行了繪圖函式也是不會顯示的。因為多數情況下視窗重繪的產生大多是因為視窗部分被遮擋或者視窗有滾動發生,改變的區域並不是整個圖形而只有一小部分,這一部分需要改變的就是pdc中的裁剪區了。因為顯示(往記憶體或者視訊記憶體都叫顯示)比繪圖過程的計算要費時得多,有了裁剪區後顯示的就只是應該顯示的部分,大大提高了顯示效率。但是這個裁剪區是mfc設定的,它已經為我們提高了顯示效率,在進行複雜圖形的繪製時如何進一步提高效率呢?那就只有去掉在裁剪區外的繪圖過程了。可以先用pdc->getclipbox()得到裁剪區,然後在繪圖時判斷你的圖形是否在這個區內,如果在就畫,不在就不畫。

如果你的繪圖過程不複雜,這樣做可能對你的繪圖效率不會有提高。

///  無閃爍背景圖繪製  //

// bool cstrucview::onerasebkgnd(cdc* pdc)

///  無閃爍背景圖繪製  //

注:程式中m_bmp 為相應 的 strucview.h中定義,為位圖資源類

protected:

cbitmap m_bmp;

並應加上下面段,當然,你可在任何時候加入loadbitmap 子例程,也可加入檔案資源,那由得你了!呵呵!^_^

cstrucview::cstrucview()

iOS tableView重新整理閃爍問題解決

原來思路是這樣的 圖中紅框是我們的訊息列表,是乙個tableview,每個訊息是乙個cell,每次接收到一條訊息立刻新增到資料陣列中,同時重新整理tableview,滾動到底部。原來的 是這樣寫的 chatdataarray addobject chatmodel chattable reloadd...

vc中關於螢幕閃爍問題解決的小結

因為invalidate 會導致整個視窗的圖象重畫,需要的時間比較長,而invalidaterect 僅僅重畫rect區域內的內容,所需時間會少一些。所以替換之後在很大程度上會減少閃爍。如果你確實需要改善閃爍的情況,計算乙個rect所用的時間比起重畫那些不需要重畫的內容所需要的時間要少得多。2 不要...

vuejs Angularjs 閃爍問題解決

angularjs 參考手冊 ng cloak指令用於在 angularjs 應用在載入時防止 angularjs 未載入完而出現的問題。angularjs 應用在載入時,文件可能會由於angularjs 未載入完而出現顯示 angularjs 進而會有閃爍的效果,ng cloak指令是為了防止該問...