分析提高GDI编程性能的方法

 
 
 
  1. IntPtrhandle=font.ToHfont();//性能瓶颈  
  2. //…  
  3. SafeNativeMethods.DeleteObject(handle); 

由于该控件在使用GDI画字时,通过调用Font.ToHfont()方法获得Font的Handle。而这个方法非常慢。并且控件在画每个Item时都被调用这个方法,Form中又有很多个这样的控件,因此调用次数相当可观。这就造成了这个性能瓶颈。

成都创新互联专业为企业提供东方网站建设、东方做网站、东方网站设计、东方网站制作等企业网站建设、网页设计与制作、东方企业网站模板建站服务,10年东方做网站经验,不只是建网站,更提供有价值的思路和整体网络服务。

由于操作系统是不允许GDI的Handle个数大于9999的。如果大于9999个的话,程序就会崩掉。因此,我们绝对不能使程序中GDI的Handle个数与某些因素有线性增长关系。所有,一般都是在使用GDI画字时创建Handle,用完之后就删除掉。这样也可以防止GDI泄露。

考虑到很多时候,Font都是相同的,如果能将Font创建的Handle缓存起来,性能就会有很大的提升。但是,缓存的Handle不及时删除的话,如果Font不相同的太多,就有机会达到操作系统允许的***个数,从而使程序崩溃。

以下是我的提高GDI编程性能解决方案:

1,使用SafeFontHandle类来防止GDI泄露。SafeFontHandle派生自SafeHandleZeroOrMinusOneIsInvalid,而SafeHandleZeroOrMinusOneIsInvalid又派生自CriticalFinalizerObject。GC会对CriticalFinalizerObject做特别处理,保证所有关键终止代码都有机会执行。

 
 
 
  1. Code  
  2. #regionTheSafeFontHandleclass  
  3.  
  4. internalsealedclassSafeFontHandle:SafeHandleZeroOrMinusOneIsInvalid  
  5. {  
  6. privateSafeFontHandle()  
  7. :base(true)  
  8. {  
  9. }  
  10.  
  11. publicSafeFontHandle(IntPtrpreexistingHandle,boolownsHandle)  
  12. :base(ownsHandle)  
  13. {  
  14. base.SetHandle(preexistingHandle);  
  15. }  
  16.  
  17. protectedoverrideboolReleaseHandle()  
  18. {  
  19. returnSafeNativeMethods.DeleteNativeFontHandle(base.handle);  
  20. }  
  21. }  
  22. #endregion 

2,使用HandleCollector类防止Font的Handle超过操作系统***限制。HandleCollector会跟踪Font的Handle,并在其达到指定阀值时强制执行垃圾回收。垃圾回收后,SafeFontHandle会释放Font的handle。

 
 
 
  1. Code  
  2. [SuppressUnmanagedCodeSecurity]  
  3. internalstaticclassSafeNativeMethods  
  4. {  
  5. privatestaticHandleCollectorFontHandleCollector=newHandleCollector("GdiFontHandle",500,1000);  
  6.  
  7. internalstaticIntPtrCreateNativeFontHandle(Fontfont)  
  8. {  
  9. IntPtrhandle=font.ToHfont();  
  10. if(handle!=IntPtr.Zero)  
  11. {  
  12. FontHandleCollector.Add();  
  13. }  
  14. returnhandle;  
  15. }  
  16.  
  17. internalstaticboolDeleteNativeFontHandle(IntPtrhandle)  
  18. {  
  19. boolsuccess=DeleteObject(handle);  
  20. if(success)  
  21. {  
  22. FontHandleCollector.Remove();  
  23. }  
  24. returnsuccess;  
  25. }  
  26.  
  27. [System.Runtime.InteropServices.DllImportAttribute("gdi32.dll")]  
  28. internalstaticexternboolDeleteObject(System.IntPtrgdiObject);  

3,使用弱引用缓存类WeakReferenceCachePool 来缓存SafeFontHandle,这样可以不影响SafeFontHandle被GC正常垃圾回收,从而释放Font的Handle。关于弱引用缓存类WeakReferenceCachePool ,可以参考《一个弱引用缓存类》这篇文章。

 
 
 
  1. Code  
  2. internalstaticclassSafeFontHandleFactory  
  3. {  
  4. #regionInstanceData  
  5. privatestaticWeakReferenceCachePool_cachePool=newWeakReferenceCachePool();  
  6. #endregion  
  7. #regionMethods  
  8. publicstaticSafeFontHandleCreateSafeFontHandle(Fontfont)  
  9. {  
  10. if(font==null)  
  11. {  
  12. thrownewArgumentNullException();  
  13. }  
  14. SafeFontHandlesafeFontHandle=_cachePool[font];  
  15. if(safeFontHandle==null)  
  16. {  
  17. IntPtrnativeHandle=SafeNativeMethods.CreateNativeFontHandle(font);  
  18. safeFontHandle=newSafeFontHandle(nativeHandle,true);  
  19. _cachePool[font]=safeFontHandle;  
  20. }  
  21. returnsafeFontHandle;  
  22. }  
  23. #endregion  

这样就成功的缓存了GDI的Handle,而且在使用完成后,GDI的Handle不会线性增长,只要有GC回收发生,GDI的Handle都会清零,或者总个数达到HandleCollector指定的阀值时,也会清零。利用GC垃圾回收机制,在性能和内存占用之间自动平衡。

这里是测试代码,提高GDI编程性能测试如下:

不使用弱引用缓存

TimeElapsed:350ms
CPUCycles:952,061,115
Gen0:1
Gen1:0
Gen2:0
GDIincrement:0

提高GDI编程性能,使用弱引用缓存

TimeElapsed:42ms
CPUCycles:142,020,499
Gen0:0
Gen1:0
Gen2:0
GDIincrement:0

【编辑推荐】

.NET图像编程(GDI+)自制“小屏保”

C# GDI+编程之基础篇

VC++.NET的GDI+入门之位图和图标

C#中使用GDI 让网站新闻标题个性化

VC中使用Gdi+合并jpg图片

文章题目:分析提高GDI编程性能的方法
文章链接:http://www.csdahua.cn/qtweb/news7/298907.html

网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网