双缓冲技术

理解:通常我们进行绘图时,都是直接使用CDC(CPaintDC等)对象在屏幕上绘制图形,如我们经常在OnPaint函数中使用:

CPaintDC dc(this);

dc.Rectangle(0,0,100,100);

在屏幕上绘制矩形。

但如果应用需要频繁的绘制图形,比如每秒要在屏幕上绘制1000个矩形。因此每绘制一个矩形到屏幕上显示,由于屏幕频繁的更新操作会导致闪烁现象。如何解决呢?使用双缓冲技术吧!

双缓冲技术的原理很简单,它通过创建一个基于内存CDC对象,该对象不会将图像直接显示在屏幕上。然后所有的绘图操作都由该对象来进行绘制(如1000个矩形)。最后将该对象绘制的图像拷贝到用于显示CDC对象上,进行屏幕显示。双缓冲技术相对于将图像的绘制和显示分离了。

实验源码:(使用基于单文档的工程)

为CbbView类添加变量:

  1. CDC* m_pMemDC;  
  2. CBitmap* m_pBitmap;  
CDC* m_pMemDC;
CBitmap* m_pBitmap;
初始化:

  1. CbbView::CbbView()  
  2. {  
  3.     m_pMemDC = new CDC();  
  4.     m_pBitmap = new CBitmap();  
  5. }  
  6. CbbView::~CbbView()  
  7. {  
  8.     delete m_pMemDC;  
  9.     delete m_pBitmap;  
  10. }  
CbbView::CbbView()
{
	m_pMemDC = new CDC();
	m_pBitmap = new CBitmap();
}
CbbView::~CbbView()
{
	delete m_pMemDC;
	delete m_pBitmap;
}
创建:

  1. int CbbView::OnCreate(LPCREATESTRUCT lpCreateStruct)  
  2. {  
  3.     if (CScrollView::OnCreate(lpCreateStruct) == -1)  
  4.         return -1;  
  5.     int width = GetSystemMetrics(SM_CXSCREEN);  
  6.     int height = GetSystemMetrics(SM_CYSCREEN);   
  7.     CDC* pDC = GetDC();  
  8.     m_pMemDC->CreateCompatibleDC(pDC);  
  9.     m_pBitmap->CreateCompatibleBitmap(pDC, width, height);  
  10.     m_pMemDC->SelectObject(m_pBitmap);  
  11.     ReleaseDC(pDC);  
  12.     CBrush brush;  
  13.     brush.CreateStockObject(WHITE_BRUSH);  
  14.     CBrush* pOldBrush = m_pMemDC->SelectObject(&brush);  
  15.     m_pMemDC->PatBlt(0, 0, width, height, PATCOPY);  
  16.     m_pMemDC->SelectObject(pOldBrush);  
  17.     return 0;  
  18. }  
int CbbView::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
	if (CScrollView::OnCreate(lpCreateStruct) == -1)
		return -1;
	int width = GetSystemMetrics(SM_CXSCREEN);
	int height = GetSystemMetrics(SM_CYSCREEN);	
	CDC* pDC = GetDC();
	m_pMemDC->CreateCompatibleDC(pDC);
	m_pBitmap->CreateCompatibleBitmap(pDC, width, height);
	m_pMemDC->SelectObject(m_pBitmap);
	ReleaseDC(pDC);
	CBrush brush;
	brush.CreateStockObject(WHITE_BRUSH);
	CBrush* pOldBrush = m_pMemDC->SelectObject(&brush);
	m_pMemDC->PatBlt(0, 0, width, height, PATCOPY);
	m_pMemDC->SelectObject(pOldBrush);
	return 0;
}
拷贝图像用于显示,在OnDraw函数中添加代码:

  1. CRect rect;  
  2. GetClientRect(&rect);  
  3. pDC->BitBlt(0, 0, rect.Width(), rect.Height(), m_pMemDC, 0, 0, SRCCOPY);  
CRect rect;
GetClientRect(&rect);
pDC->BitBlt(0, 0, rect.Width(), rect.Height(), m_pMemDC, 0, 0, SRCCOPY);
响应单击事件,进行绘图:

  1. void CbbView::OnLButtonDown(UINT nFlags, CPoint point)  
  2. {     
  3.     m_pMemDC->TextOutW(point.x, point.y, L"you are not alone");  
  4.     CBrush brush(RGB(0, 0, 255));  
  5.     m_pMemDC->FillRect(CRect(CPoint(point.x,point.y+20),CSize(100,100)),&brush);  
  6.     Invalidate(FALSE);  
  7.     CScrollView::OnLButtonDown(nFlags, point);  
  8. }  
void CbbView::OnLButtonDown(UINT nFlags, CPoint point)
{	
	m_pMemDC->TextOutW(point.x, point.y, L"you are not alone");
	CBrush brush(RGB(0, 0, 255));
	m_pMemDC->FillRect(CRect(CPoint(point.x,point.y+20),CSize(100,100)),&brush);
	Invalidate(FALSE);
	CScrollView::OnLButtonDown(nFlags, point);
}
无论单击多快·,都不会闪烁。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值