GDI使用经验总结

        本人由于工作的原因,对GDI的研究较多,下面是我在使用中的一些经验总结,希望大家有问题的指出问题,如果能对初学者有一点点帮助,我也就算是没有白写此文了!

        windows内部对于GDI的操作赋于了很高的权限,但要想做一个好的图形界面,并且占用资源少的图形界面的话,还是要下一番工夫的,下面我总结出来几条:

一:
双缓冲,这是耳朵听起老茧的东西的,这其中,主要是建立内存兼容DC和内存兼容位图,如下:
HDC hDC = ::GetDC(m_hWnd);
hFrceDC = CreateCompatibleDC(hDC); //内存兼容DC
hFrceBmp = CreateCompatibleBitmap(hDC, WinWidth, WinHeight); //内存兼容位图
SelectObject(hFrceDC, hFrceBmp); //选入内存兼容DC
::ReleaseDC(m_hWnd, hDC);

以后所有的GDI操作,比如LineTo等,都只对hFrceDC,当要刷新的时候:
void CST_CurveCtrl::OnDraw(CDC* pdc, const CRect& rcBounds, const CRect& rcInvalid)
{
 BitBlt(pdc->m_hDC, rcInvalid.left, rcInvalid.top, rcInvalid.Width(), rcInvalid.Height(), hFrceDC, rcInvalid.left, rcInvalid.top, SRCCOPY);
 ...
}

简单吧,这就是为什么双缓冲是GDI编程中最为基本的技巧了。这样不管你的绘制有多少的复杂,屏幕不会再有闪烁,绘制的复杂只会占用CPU多一点而已。

二:
刷新区域很重要,拿上面的OnDraw函数来说,千万不要去刷新rcBounds区域,它是整个客户区域,而应该刷新rcInvalid,它才是无效区域。如果对rcBounds区域刷新,当然表现出来的效果没什么分别,只是会占用更多的CPU。

三:
是否刷新背景很重要,有些新手将上面的双缓冲用到对话框之上,然后告诉我,还是闪烁!那是因为刷了背景的原因,因为用BitBlt绘制的东西,在windows看来,它不是一个窗口,它不像一个按钮,因为按钮是一个窗口,所以windows在刷背景的时候,会从刷新区域里面减掉按钮所占的区域,那么回到正题,在对话框上BitBlt的时候,如果位图显示占满了对话框的整个客户区,这就简单不过了,直接响应WM_ERASEBKGND消息,然后返回TRUE:
BOOL CXXX::OnEraseBkgnd(CDC* pDC)
{
 return TRUE;
}
如果BitBlt不会占满整个客户区,那么很抱歉,有点麻烦了,你要生成一个刚好不包括BitBlt显示区域的区域,然后对DC执行SelectClipRgn函数,请在MSDN里面搜索“Region Functions”,里面的函数大多你可能都用得上,主要是创建区域,操作区域(比如求AND,求OR,求DIFF等)。

四:
GDI函数调用次数很重要,用上双缓冲和消除刷新背景之后,闪烁的问题可以认为是解决了,下面我们将讨论减少CPU使用率的问题。GDI操作是比较浪费CPU资源的,比如频率的调用GDI函数,势必浪费CPU资源,这其中又特别是字符的打印操作,为此GDI提供PolyTextOut函数,调用它一次,可以输出任意多条的字符,每一条都是独立设置输出位置的,这个函数的使用场合我举个例子,比如你在绘制一个坐标上的刻度值,那么这个函数再好不过了。与此功能类似的还有PolylineTo、Polyline、PolyPolyline、Polygon、PolyPolygon等函数。

五:

注意ExtTextOut函数,这个函数表面上看是用来显示字符串的,但它还可以用来绘制一个填充的矩形,无边框的,作用和FillSolidRect一样,在此种情况下,千万不要使用Rectangle函数(选一个空画笔也能达到无边框效果),ExtTextOut函数速度最快我可以找到证据,请看MFC源码:
void CDC::FillSolidRect(LPCRECT lpRect, COLORREF clr)
{
 ASSERT_VALID(this);
 ASSERT(m_hDC != NULL);

 ::SetBkColor(m_hDC, clr);
 ::ExtTextOut(m_hDC, 0, 0, ETO_OPAQUE, lpRect, NULL, 0, NULL);
}
在wingdix.cpp里面。

六:
区域刷新技术,这个技术我不知道大家是不是这样叫的,如果不是这样叫的,就是我给它以的名字,它的思想就是,将客户区分为若干个区域(一般是矩形就行了,速度快),具体怎么分,分多少个没有定论,理论上,可能需要单独刷新的地方,就分为一个区域,每当这个区域需要刷新的时候,只对这个区域进行绘制,并且BitBlt到屏幕上。具体来说,每一个区域应该对应一个刷新函数(或者一个switch语句的某一个case项),这个函数执行GDI操作,当这个区域需要刷新的时候,调用这个函数将新的东西绘制到一个内存兼容DC上,然后让这个矩形失效,那么windows会在适当的时候,调用OnDraw(或类似的函数),在OnDraw里面,直接用BitBlt对无效区域(就是前面提到的那个矩形区域)进行贴图即可。

这个技术在我的ST_Curve用得比较多,下面是一个截图,更多详细情况请看我的主页:

www.st-curve.cn

  • 2
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 10
    评论
Visual C++6.0使用GDI+的一般方法 1. 载解压GDI+开发包; 2. 正确设置include & lib 目录; 3. stdafx.h 添加: #ifndef ULONG_PTR #define ULONG_PTR unsigned long* #endif #include 4. 程序中添加GDI+的包含文件gdiplus.h以及附加的类库gdiplus.lib。 通常gdiplus.h包含文件添加在应用程序的stdafx.h文件中,而gdiplus.lib可用两种进行添加: 第一种是直接在stdafx.h文件中添加下列语句: #pragma comment( lib, "gdiplus.lib" ) 另一种方法是: 在VC.net中添加库文件在:项目菜单->属性->链接器->输入 举个例子: (1)在应用程序项目的应用类中,添加一个成员变量,如下列代码: ULONG_PTR m_gdiplusToken; 其中,ULONG_PTR是一个DWORD数据类型,该成员变量用来保存GDI+被初始化后在应用程序中的GDI+标识,以便能在应用程序退出后,引用该标识来调用Gdiplus:: GdiplusShutdown来关闭GDI+。 (2)在应用类中添加ExitInstance的重载,并添加下列代码用来关闭GDI+: int CGDITestApp::ExitInstance() { Gdiplus::GdiplusShutdown(m_gdiplusToken); return CWinApp::ExitInstance(); } (3)在应用类的InitInstance函数中添加GDI+的初始化代码: 注意:下面这些GDI+的初始化代码必须放在m_pMainWnd->UpdateWindow();之前。 CWinApp::InitInstance(); Gdiplus::GdiplusStartupInput gdiplusStartupInput; Gdiplus::GdiplusStartup(&m_gdiplusToken, &gdiplusStartupInput, NULL); (4)在需要绘图的窗口或视图类中添加GDI+的绘制代码。 下面分别就单文档和基于对话框应用程序为例,说明使用GDI+的一般过程和方法。 1. 在单文档应用程序中使用GDI+ 在上面的过程中,我们就是以一个单文档应用程序Ex_GDIPlus作为示例的。下面列出第4步所涉及的代码: void CGDITestView::OnDraw(CDC* pDC) { CGDITestDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here usingnamespace Gdiplus; Graphics graphics(pDC->m_hDC); Pen newPen(Color(255,0,0),3); HatchBrush newBrush(HatchStyleCross,Color(255,0,255,0),Color(255,0,0,255));//创建一个填充画刷,前景色为绿色,背景色为蓝色 graphics.DrawRectangle(&newPen,50,50,100,60);// 在(50,50)处绘制一个长为100,高为60的矩形 graphics.FillRectangle(&newBrush,50,50,100,60); // 在(50,50)处填充一个长为100,高为60的矩形区域 } 编译并运行,结果如图:

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值