gdi编程中的资源泄露

转载 2012年03月27日 10:01:54
在windows系列上做编程,gdi是一个很重要的技术点,有很多程序在运行多次后出现异常,除了众所周知的内存泄露以外,gdi资源泄露也是一个很直接的原因.今天就把我自己在编程中总结的一些经验给大家分享,欢迎高手补充.
1.Create出来的gdi对象,一定要用DeleteObject来释放,释放顺序是先Create的后释放,后Create的先释放.
这里的Create指的是以它为开头的gdi函数,比如,CreateDIBitmap,CreateFont等等,最后都要调用DeleteObject来释放.

2.Create出来的dc要用DeleteDC来释放,Get到的要用ReleaseDC释放.

3.确保释放DC的时候DC中的各gdi对象都不是你自己创建的;确保个gdi对象在释放的时候不被任何dc选中使用.
假如我们要使用gdi函数画图,正确的步骤应该如下:
a.创建一个内存兼容dc(CreateCompatibleDC)
b.创建一个内存兼容bitmap(CreateCompatibleBitmap)
c.关联创建的内存兼容dc和bitmap(SelectObject)
d.画图
e.BitBlt到目的dc上
f.断开内存兼容dc和bitmap关联(SelectObject)
g.销毁内存兼容bitmap
h.销毁内存兼容dc
由于SelectObject在选入一个新的gdi对象的时候会返回一个原来的gdi对象(假如成功的话),所以需要在步骤c的时候保存返回值,在步骤f的时候当作入口参数使用.还有,步骤g和步骤h实际上顺序可以随意,因为他们两个此刻已经没有关系了,但是为了结构清晰,我建议按照"先Create的后释放,后Create的先释放"的原则进行.
关于步骤f,可能会有争议,因为即使省略这一步,步骤g和步骤h看起来照样可以返回一个成功的值.但实际上可能并没有执行成功,至少boundschecker会报告有错,错误信息大致是说,在释放dc的时候还包含有非默认的gdi对象,在释放gdi对象的时候又说这个gdi对象还被一个dc在使用.所以,我建议保留步骤f.

4.关于98下使用CreateCompatibleBitmap
按照msdn的说法,创建出来的size不能超过16m.实际情况是这样吗?非也~!从我自己做的测试结果来看(win98se-sc),这个值在2044*2043和2044*2044之间,然而,后来在另外一个98系统上这个值也不行,后来我干脆把上限给成了2000*2000.很幸运,到现在还没有出问题,但我不能保证这个数字就是正确的.还有一点,假如宽或高有一个超过32768,哪怕另外一个值是1,也会创建失败,有兴趣的可以自己做个测试.如果要想保证这个函数在98下永远成功,可以试试下面的代码:
float factor = 10.f;
while(!bitmap.CreateCompatibleBitmap(&dc ,nWidth*factor ,nHeight*factor))
{
   factor -= 0.01f;  
}
这样至少可以保证宽和高是成比例的:)

5.关于在打印机上使用BitBlt
有时候在内存兼容dc里面已经做好图了,但在使用BitBlt的时候却会失败.这个时候,首先确认创建的内存兼容dc和bitmap是不是使用打印机的dc,如果确认无误,还是执行BitBlt失败,那80%可能是内存兼容bitmap太大了,请按如下方法再试试:
创建另外一个内存兼容dc2和一个比较小的内存兼容biimap2,大概是1000*1000吧,我是这样用的:)然后把dc里面的内容分成块(1000*1000),把每一块BitBlt到dc2上面,再从dc2里面BitBlt到打印dc上.有人可能会有这样的疑问:那为什么不直接把dc里面的内容分几次BitBlt到打印机上呢?有区别吗?答案是肯定的,如果dc里面的bitmap太大,哪怕你想BitBlt一个10*10的区域到打印机上都会失败.

f断开内存兼容dc和bitmap关联(SelectObject)
不会造成内存泄漏(Memory   Leak),但有可能造成资源泄漏(Resource   Leak)。
这两句话的作用只是用来恢复原来的画笔和画刷。如果你的程序是要长时间使用,并且这个函数是要反复被使用的,那么,建议你加上去.如果只是一般性的使用,加不加也无所谓.当然,删除的时候加不加都一样

windbg检测句柄泄露(定位到具体代码)

1.构造一个测试用例
  • yockie
  • yockie
  • 2014年10月30日 01:12
  • 6250

C#内存泄露与资源释放 经验总结

经验总结:C#并不是有了垃圾回收机制,你就可以对内存管理放任不管,其实在稍不注意的时候,可能就造成了内存泄露,或者超大内存占用~ 以下记录下遇到过的内存优化-内存泄露的问题与应对方案。 1...
  • Yokeqi
  • Yokeqi
  • 2014年11月13日 18:39
  • 10443

java中内存泄露有几种?如何分析泄露原因

一、Java内存回收机制 不论哪种语言的内存分配方式,都需要返回所分配内存的真实地址,也就是返回一个指针到内存块的首地址。Java中对象是采用new或者反射的方法创建的,这些对象的创建都是在堆(He...
  • zhousenshan
  • zhousenshan
  • 2016年10月19日 22:34
  • 2082

GDI对象的使用和防止资源泄露

最近的项目在绘制大量采集到的数据时,会有内存泄露的致程序崩溃的情况,反复推测后可能是GDI对象的释放问题,现在网上搜取如下经验,应该很有帮助GDI对象的使用和防止资源泄露在windows系列上做编程,...
  • yuanweihuayan
  • yuanweihuayan
  • 2011年02月23日 13:45
  • 617

Wince GDI资源泄露可能原因

Wince GDI资源泄露可能原因 在网上找了很多人的博客来看,在各种论坛里面看高手的回复,把Wince GDI资源泄露的可能原因总结了一下 (1)不要用CDC做直接用HDC (2)...
  • aristolto
  • aristolto
  • 2012年03月31日 16:39
  • 2137

如何防止GDI资源泄露

在windows系列上做编程,gdi是一个很重要的技术点,有很多程序在运行多次后出现异常,除了众所周知的内存泄露以外,gdi资源泄露也是一个很直接的原因.今天就把我自己在编程中总结的一些经验给大家分享...
  • weiqubo
  • weiqubo
  • 2011年09月28日 16:58
  • 1327

Net资源泄露(内存泄露,GDI泄露,handle 泄露等)的终极解决方案

http://blog.csdn.net/yuanhuiqiao/article/details/5264480 摘要   本文主要讨论了,什么是.Net内存泄露?如何确定是发生了...
  • king_idea8848
  • king_idea8848
  • 2012年04月17日 10:58
  • 357

WindowsNT中,GDI资源的管理

WindowsNT中,GDI资源的管理 http://blog.chinaunix.net/uid-171942-id-2819968.html   GDI资源为什么会耗尽呢,到...
  • wwww4
  • wwww4
  • 2015年02月05日 11:10
  • 391

windows资源管理(内核对象/GDI对象/user对象)

在使用c++进行windows编程时,程序员除了管理使用new/malloc动态在堆上分配出来的内存外,还需要对windows的内核对象、GDI对象和user对象进行管理,这些对象使用句柄来标识,通过...
  • fion45
  • fion45
  • 2013年08月14日 17:32
  • 1300

C# GDI+双缓冲技术 及 资源

http://blog.sina.com.cn/s/blog_4e7453df0100d2fy.html   C# GDI+双缓冲技术 http://www.webmis.com.cn/ http:...
  • mkbaoquan
  • mkbaoquan
  • 2017年12月26日 11:17
  • 21
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:gdi编程中的资源泄露
举报原因:
原因补充:

(最多只允许输入30个字)