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

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-13 09:53:29

GDI使用经验总结的相关文章

GDI+(1)--入门

GDI+入门 摘要:最近在对以前的的项目进行优化,重新做了个面板,为了达到逼真的效果,界面面很多地方需要实现透明.半透明.镂空.分层.旋转等效果.在做UI的过程中,发现了很多GDI无法处理的问题,于是我转而结合使用GDI+.本文着重于GDI+入门介绍,同时也GDI与GDI+进行了对比. GDI+与GDI一样都具有设备无关性.所以使用GDI+ 时候,我们无需考虑显示设备的具体情况(这一点跟DirectX有和大区别).GDI+将应用程序和图形硬件隔离,正是这种隔离允许我们开发与设备无关的应用程序.

GDI+ 画渐变色环

在onpaint() 函数中加入如下代码,本次利用DrawArc来实现. #define PI 3.1415926 int angle=360; int x=(rect.Width()-300)/2; int y=190; int width=300; int hight=300; int wide=34; graphics.SetSmoothingMode(SmoothingModeAntiAlias);//抗锯齿 for(float i = 0.0; i < PI; i += (float)

vs2008中使用gdi+的设置

vs2008中使用gdi+ 1.新建一个mfc工程 2.在stdafx.h文件中加入以下几行语句:#include <gdiplus.h>                //#pragma comment(lib, "gdiplus.lib") //在工程属性中添加亦可using namespace Gdiplus;            //使用GDI+的命名空间, 若不用的话每次使用Gdiplus时均加上命名空间亦可 3.修改App类在App类(以下例子中为CTestA

win32用GDI+加载png图片作为背景图

#include <windows.h> #include <gdiplus.h> /* GDI+ startup token */ ULONG_PTR gdiplusStartupToken; /* Declare Windows procedure */ LRESULT CALLBACK WindowProcedure (HWND, UINT, WPARAM, LPARAM); // UpdateLayeredWindow Defination typedef BOOL(*UP

win32学习之 --------GDI使用 代码记录

LRESULT QMainFrame::onPaint(WPARAM wParam, LPARAM lParam) { PAINTSTRUCT ps; HDC hDC = ::BeginPaint(m_hWnd, &ps); RECT rcClient; GetClientRect(&rcClient); assert(hDC); { /**画字体 DrawText(hDC,_T("test"), _tcslen(_T("test")),&r

Delphi GDI+基本用法总结

GDI+以前只是听说过,还没怎么用过,这段时间用了用,觉得挺好用的.在这里总结一下.留个备忘. GDI+(Graphics Device Interface plus)是Windows XP中的一个子系统,它主要负责在显示屏幕和打印设备输出有关信息,它是一组通过C++类实现的应用程序编程接口.由于它是图形绘制相关的接口,那先了解一下常用的几种图片格式. 常见的图片格式 常见的图片格式有JPEG, BMP, PNG. - JPEG:它用有损压缩方式去除冗余的图像或彩色数据, 获取得极高的压缩率的同

GDI基础

1.GDI 图像设备接口,windows程序显示点,线,图像,文字时,在显示器或打印机输出这些内容时,需要使用GDI2.when--在WM_PAINT时  where--设备描述表  how--调用GDI函数:  case WM_PAINT:  HDC hdc; //设备描述变量  PAINTSTRUCT ps; //绘图信息结构变量  hdc = BeginPaint(hwnd, &ps); //赋值给hdc  GetClientRect(hwnd, &rect);   //TextOu

Delphi中使用GDI+进行绘图(1)

Delphi的VCL类库中,默认使用的是GDI绘图接口,该接口封装了Win32 GDI接口,能够满足基本的绘图功能,但如果要实现更高级的绘图功能,往往比较困难,GDI+是微软在GDI之后的一个图形接口,功能比GDI丰富很多,在VCL中使用GDI+,能够实现很多高级绘图功能. 目前有多种Delphi对GDI+的封装实现,以下介绍最简单的两种: 1.使用Delphi内置的GDI+接口 首先,创建一个VCL Form应用,窗口文件如下: Pascal Code 12345678910111213141

VB6 GDI+ 入门教程——图片

版权声明:转载时请以超链接形式标明文章原始出处和作者信息及本声明http://vistaswx.blogbus.com/logs/41225905.html VB自己的绘图语句都需要用LoadPicture载入图片,同样,GDI+中也需要. 1.载入(初始化)图片资源 (1)来自文件:GdipLoadImageFromFile 我们先来看看这个最简单基本的载入图片来自文件: [vb] view plain copy print? Dim img As Long, img_W As Long, i