cWinForm利用GDI的双缓冲技术来提高绘图效率

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

C# WinForm利用GDI+的双缓冲技术来进步绘图效率
前言
进入.NET时代,Windows的绘图技术也从GDI晋级到了GDI+,从名字就能知道GDI+是对以前传统GDI 绘图技术的一次晋级,不过在微软几乎把所有的新技术都冠之.NET的情况下,GDI+竟然不叫做,还真让我感到有点意外了。

:〕
GDI+在一种与设备无关的环境下提供了一套统一的绘图编程模型,极大的进步了Windows绘图编程的方便性,我们再也不用创立什么各种各样复杂的设备环境了,说实话,我如今想起来都头疼。

题归正传,关于如何进展GDI+的根本编程,我不能过多的加以描绘,假如有对此概念还不太清楚的朋友,建议先去理解一下相关的资料,我们在这里主要讨论的是一种进步绘图效率〔主要是动画效率〕的双缓冲技术在GDI+中的应用和实现。

实现目的
为了能清楚的比照应用双缓冲技术前后的效果,我编写了一段程序来进展测试。

首先,我创立了一个普通的Windows Application,在主Form中,我放置了一个定时器:timer1,然后将它的Interval属性设置为10,然后在Form上放置两个按纽,分别用来控制定时器的开启和关闭,最后,我还放置了一个label控件,用来显示绘图的帧数。

测试程序
在timer1的timer1_Tick事件中,我写下了如下的代码(其中flag是一个bool型标志变量):DateTime t1 = DateTime.Now;
Graphics g = this.CreateGraphics();
if(flag)
{
brush = new LinearGradientBrush(new PointF(0.0f, 0.0f),
new PointF(700.0f, 300.0f), Color.Red, Color.Blue);
flag = false;
}
else
{
brush = new LinearGradientBrush(new PointF(0.0f, 0.0f),
new PointF(700.0f, 300.0f), Color.Blue, Color.Red);
flag = true;
}
for(int j = 0; j < 60; j ++)
{
for(int i = 0; i < 60; i++)
{
g.FillEllipse(brush, i * 10, j * 10, 10, 10);
}
}
DateTime t2 = DateTime.Now;
TimeSpan sp = t2 - t1;
float per = 1000 / liseconds;
运行后,我点击“开场〞按纽,效果如下列图所示:
应用双缓冲以前的效果图〔帧数:5帧/秒〕
正如大家所看到的,我在程序中使用循环画了几百个圆形,然后在每次的定时器脉冲事件中使用不同方向的线性渐变来对它们进展填充,形成了一个动画效果。

不过不幸的是,程序运行起来闪烁很严重,几乎每次刷新的时候都可以看到一条很明显的扫描线从上渐渐的刷到下来完成整幅图形的刷新动作。

假如你不是要模拟老式雷达的区域扫描的话,这种速度不会满足你的要求。

改良代码
下面是我改良以后的代码:
DateTime t1 = DateTime.Now;
Bitmap bmp = new Bitmap(600, 600);
Graphics g = Graphics.FromImage(bmp);
if(flag)
{
brush = new LinearGradientBrush(new PointF(0.0f, 0.0f),
new PointF(700.0f, 300.0f), Color.Red, Color.Blue);
flag = false;
}
else
{
brush = new LinearGradientBrush(new PointF(0.0f, 0.0f),
new PointF(700.0f, 300.0f), Color.Blue, Color.Red);
flag = true;
}
for(int j = 0; j < 60; j ++)
{
for(int i = 0; i < 60; i++)
{
g.FillEllipse(brush, i * 10, j * 10, 10, 10);
}
}
this.CreateGraphics().DrawImage(bmp, 0, 0);
DateTime t2 = DateTime.Now;
TimeSpan sp = t2 - t1;
float per = 1000 / liseconds;
运行后,我点击“开场〞按纽,效果如下列图所示:
应用双缓冲以后的效果图〔帧数:9帧/秒〕
经过改良后,画面刷新速度大大加快,绝对看不到任何的“扫描线〞,帧数也从5帧一下就进步到了9帧,几乎是两倍于前的速度。

这终究是什么原因呢?让我来讲述其中的道理。

因为圆是要一个一个画上去,所以每画一个圆,系统就要做一次图形的绘制操作,图形的重绘是很占用资源的,当需要重绘的图形数量很多的时候,所造成的系统开销就特别大,造成我们看到的那种刷新缓慢的情况。

那么如何来解决这个问题呢?
答案就是双缓冲,何谓“双缓冲〞?它的根本原理就是:先在内存中开拓一块虚拟画布,然后将所有需要画的图形先画在这块“虚拟画布〞上,最后在一次性将整块画布画到真正的窗体上。

因为所有的单个图形的绘制都不是真正的调用显示系统来“画〞,所以不会占用显示系统的开销,极大的进步的绘图效率。

实现双缓冲的详细步骤
我再来详细解释一下刚刚实现双缓冲的详细步骤:
1、在内存中建立一块“虚拟画布〞:
Bitmap bmp = new Bitmap(600, 600);
2、获取这块内存画布的Graphics引用:
Graphics g = Graphics.FromImage(bmp);
3、在这块内存画布上绘图:
g.FillEllipse(brush, i * 10, j * 10, 10, 10);
4、将内存画布画到窗口中
this.CreateGraphics().DrawImage(bmp, 0, 0);。

相关文档
最新文档