【学习笔记】Windows GDI绘图(十二)双缓冲管理(用GIF动画测试)

2024-06-07 18:36

本文主要是介绍【学习笔记】Windows GDI绘图(十二)双缓冲管理(用GIF动画测试),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 引言
  • 默认双缓冲
    • SetStyle
  • 手动管理双缓冲图形
    • BufferedGraphicsManager缓冲图形管理器
    • BufferedGraphicsContext 缓冲图形上下文
    • BufferedGraphics 图形缓冲区
    • 验证双缓冲的效果(Gif动画显示非正常速度)
    • 结束语
      • 性能对比

引言

图形编程中一个常见的问题就是闪烁,当需要绘制多个复杂的图形时可能导致渲染的图像出现闪烁或其他不可接受的形式。

.Net中提供了双缓冲(Double buffering)来解决这一问题。

双缓冲使用内存缓冲区来解决多个绘制操作相关的闪烁问题。启用双缓冲后,所有的绘制操作都会首先渲染到内存的缓冲区,到所有绘制操作完成后,再直接复制到与其关联的绘图表面上。由于在屏幕上仅执行一次图形操作,因此消除了闪烁等问题

默认双缓冲

在.Net中要启用控件的默认双缓冲功能,只需将设置DoubleBuffered为True或调用SetStyle方法。

SetStyle

原型:

protected void SetStyle (System.Windows.Forms.ControlStyles flag, bool value);

ControlStyles枚举

说明
AllPaintingInWmPaint如果为 true,则控件忽略窗口消息 WM_ERASEBKGND 以减少闪烁。
仅当将 UserPaint 位设置为 true 时,才应用此样式。
DoubleBuffer如果为 true,则在缓冲区中进行绘制,并且完成后将结果输出到屏幕。 双缓冲可以防止因重绘控件而引起的闪烁。
如果将 DoubleBuffer 设置为 true,则还应将 UserPaint 和 AllPaintingInWmPaint 设置为 true。
UserPaint如果为 true,则会由控件而不是由操作系统来绘制控件自身。
如果 false,则不会引发 Paint 事件。 此样式仅适用于从 Control 派生的类。
OptimizedDoubleBuffer如果为 true,则控件将首先绘制到缓冲区而不是直接绘制到屏幕,这可以减少闪烁。
如果将此属性设置为 true,则还应将 AllPaintingInWmPaint 设置为 true
ResizeRedraw如果为 true,则控件会在调整大小时进行重绘。

除了设置DoubleBuffered = True,还可以

   this.SetStyle(ControlStyles.OptimizedDoubleBuffer | ControlStyles.UserPaint | ControlStyles.AllPaintingInWmPaint,true);this.UpdateStyles();

手动管理双缓冲图形

手动管理双缓冲图形,需要用到BufferedGraphics和

BufferedGraphicsManager缓冲图形管理器

原型:

public static class BufferedGraphicsManager

作用:提供对应用程序域的主缓冲图形上下文对象的访问。

BufferedGraphicsManager类允许你实现自定义双缓冲。一般是获取其属性Current为 BufferedGraphicsContext类型。

BufferedGraphicsContext 缓冲图形上下文

原型:

public sealed class BufferedGraphicsContext : IDisposable

作用:提供创建可用于双缓冲图形缓冲区的方法。

1、通过 BufferedGraphicsManager.Current获取到一个BufferedGraphicsContext 对象
2、设置MaximumBuffer为最大缓冲区大小(一般是显示图像的宽+1和高+1,缓冲区大小的内存为被长期占用,而图像大小缓冲区大小时会被临时使用,直到该对象被释放)。
3、调用Allocate方法,分配一个指定大小的缓冲区图形(BufferedGraphics)

context = BufferedGraphicsManager.Current;
context.MaximumBuffer = new Size(gifImage.Width + 1, gifImage.Height + 1);
bufferdGraphics = context.Allocate(e.Graphics, new Rectangle(0, 0, gifImage.Width, gifImage.Height));

BufferedGraphics 图形缓冲区

原型:

public sealed class BufferedGraphics : IDisposable

作用:提供用于双缓冲的图形缓冲区。

1、其属性Graphics可用于绘制图形
2、再调用Render方法,绘制到指定的Graphics上。

验证双缓冲的效果(Gif动画显示非正常速度)

通过鼠标点击窗体控制切换不同的效果,分别是

  • 禁用双缓冲模式
  • 启用默认双缓冲模式
  • 自定义管理双缓冲模式
    先看看不同方式下的效果
    双缓冲
    1、不启用双缓冲时,屏幕严重闪烁
    2、启用双缓冲时,默认和自定义效果(不知使用方法上有问题)差不多。
    3、注意,这个是用来测试绘制效果的,实际的Gif播放不是这样使用。
    public partial class FrmDoubleBufferd : Form{public FrmDoubleBufferd(){InitializeComponent();}private void FrmDoubleBufferd_Load(object sender, EventArgs e){Init();}/// <summary>/// 双缓冲类型/// 0-禁用双缓冲/// 1-默认双缓冲/// 2-手管理双缓冲/// </summary>private int BufferdType = -1;private void FrmDoubleBufferd_Click(object sender, EventArgs e){BufferdType++;if (BufferdType > 2) BufferdType = 0;if (BufferdType == 0){this.DoubleBuffered = false;this.SetStyle(ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint, false);}else{this.DoubleBuffered = true;this.SetStyle(ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint | ControlStyles.UserPaint, true);}CurrFrameIndex = 0;TotalCount = 0;elaspedTime = new ConcurrentBag<(long, long)>();dt = DateTime.Now;NextGifImage();}private int CurrFrameIndex = 0;private int FrameCount = 0;FrameDimension Dimension;private Image GifImage = null;private Image AIWoman = null;Rectangle srcRect;private BufferedGraphicsContext context;private BufferedGraphics bufferdGraphics;private void Init(){GifImage = Image.FromFile("Heartbeat.gif");AIWoman = Image.FromFile("AIWoman.png");srcRect = new Rectangle(0, 0, AIWoman.Width, AIWoman.Height);if (ImageAnimator.CanAnimate(GifImage)){Dimension = new FrameDimension(GifImage.FrameDimensionsList[0]);FrameCount = GifImage.GetFrameCount(Dimension);}context = BufferedGraphicsManager.Current;context.MaximumBuffer = new Size(GifImage.Width + 1, GifImage.Height + 1);bufferdGraphics = context.Allocate(this.CreateGraphics(), new Rectangle(0, 0, GifImage.Width, GifImage.Height));NextGifImage();}Random random = new Random((int)DateTime.Now.Ticks);private void FrmDoubleBufferd_Paint(object sender, PaintEventArgs e){var dstRect = new Rectangle(random.Next(100, 600), random.Next(10, 600), 400, 400);e.Graphics.DrawImage(GifImage, 0, 0, GifImage.Width, GifImage.Height);e.Graphics.DrawImage(AIWoman, dstRect, srcRect, GraphicsUnit.Pixel);if (BufferdType == 0){e.Graphics.DrawString($"禁用双缓冲模式", Font, Brushes.Red, new PointF(20, 20));DrawElaspedTime(e.Graphics);}else if (BufferdType == 1){e.Graphics.DrawString($"默认双缓冲模式", Font, Brushes.Red, new PointF(20, 20));DrawElaspedTime(e.Graphics);}else{return;}//绘制结束后,显示下一帧NextGifImage();}ConcurrentBag<(long, long)> elaspedTime = new ConcurrentBag<(long, long)>();private void DrawElaspedTime(Graphics g){int height = 40;foreach (var time in elaspedTime){g.DrawString($"{time.Item1},耗时:{time.Item2}", Font, Brushes.Red, new PointF(20, height));height += 20;}}private long TotalCount= 0;DateTime dt = DateTime.Now;bool redraw= false;//切换Gif显示帧private void NextGifImage(){do{TotalCount++;CurrFrameIndex++;if (CurrFrameIndex >= FrameCount){CurrFrameIndex = 0;}if (CurrFrameIndex % 50 == 0){var elasped = (CurrFrameIndex, (long)(DateTime.Now - dt).TotalMilliseconds);elaspedTime.Add(elasped);dt = DateTime.Now;}//更新为下一帧GifImage.SelectActiveFrame(Dimension, CurrFrameIndex);ImageAnimator.UpdateFrames(GifImage);if (BufferdType == 2){//自定义管理双缓冲时,不在Paint事件中更新var dstRect = new Rectangle(random.Next(100, 600), random.Next(10, 400), 400, 400);bufferdGraphics.Graphics.Clear(Color.White);bufferdGraphics.Graphics.DrawImage(GifImage, 0, 0, GifImage.Width, GifImage.Height);bufferdGraphics.Graphics.DrawImage(AIWoman, dstRect, srcRect, GraphicsUnit.Pixel);bufferdGraphics.Graphics.DrawString($"自定义管理双缓冲模式", Font, Brushes.Red, new PointF(20, 20));DrawElaspedTime(bufferdGraphics.Graphics);bufferdGraphics.Render();Application.DoEvents();redraw = true;}else{//触发重绘this.Invalidate();redraw = false;}}while(redraw);            }private void FrmDoubleBufferd_FormClosing(object sender, FormClosingEventArgs e){bufferdGraphics.Dispose();GifImage.Dispose();AIWoman.Dispose();}}

结束语

性能对比

  1. 自定义管理双缓冲
    在这里插入图片描述
  2. 默认双缓冲
    在这里插入图片描述
    左上角的数字意义是,每绘制50帧(绘制Gif的同时还绘制了美女)的耗时(单位ms)。

原本想测试下,自定义管理双缓冲会不会比默认启动的双缓冲性能要高,不知是使用的方法不问题,还是其他什么原因,没感觉到有更高的性能(反而觉得慢了些),如果您有更好的例子可以说明,麻烦留言,万分感谢。

再次强调,实际的Gif动画播放不是像本文那样实现的,可通过ImageAnimator类的相关方法实现,本文之所以这样写,本是想着每绘制完一帧后,开始处理下一帧,看哪种方法更快。

在未完全吃透自定义管理双缓冲情况,建议还是用默认的双缓冲就可以了。

这篇关于【学习笔记】Windows GDI绘图(十二)双缓冲管理(用GIF动画测试)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1039971

相关文章

Linux创建服务使用systemctl管理详解

《Linux创建服务使用systemctl管理详解》文章指导在Linux中创建systemd服务,设置文件权限为所有者读写、其他只读,重新加载配置,启动服务并检查状态,确保服务正常运行,关键步骤包括权... 目录创建服务 /usr/lib/systemd/system/设置服务文件权限:所有者读写js,其他

Linux挂载linux/Windows共享目录实现方式

《Linux挂载linux/Windows共享目录实现方式》:本文主要介绍Linux挂载linux/Windows共享目录实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录文件共享协议linux环境作为服务端(NFS)在服务器端安装 NFS创建要共享的目录修改 NFS 配

基于Python开发Windows自动更新控制工具

《基于Python开发Windows自动更新控制工具》在当今数字化时代,操作系统更新已成为计算机维护的重要组成部分,本文介绍一款基于Python和PyQt5的Windows自动更新控制工具,有需要的可... 目录设计原理与技术实现系统架构概述数学建模工具界面完整代码实现技术深度分析多层级控制理论服务层控制注

在Node.js中使用.env文件管理环境变量的全过程

《在Node.js中使用.env文件管理环境变量的全过程》Node.js应用程序通常依赖于环境变量来管理敏感信息或配置设置,.env文件已经成为一种流行的本地管理这些变量的方法,本文将探讨.env文件... 目录引言为什么使php用 .env 文件 ?如何在 Node.js 中使用 .env 文件最佳实践引

Oracle数据库在windows系统上重启步骤

《Oracle数据库在windows系统上重启步骤》有时候在服务中重启了oracle之后,数据库并不能正常访问,下面:本文主要介绍Oracle数据库在windows系统上重启的相关资料,文中通过代... oracle数据库在Windows上重启的方法我这里是使用oracle自带的sqlplus工具实现的方

python库pydantic数据验证和设置管理库的用途

《python库pydantic数据验证和设置管理库的用途》pydantic是一个用于数据验证和设置管理的Python库,它主要利用Python类型注解来定义数据模型的结构和验证规则,本文给大家介绍p... 目录主要特点和用途:Field数值验证参数总结pydantic 是一个让你能够 confidentl

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版

Redis实现高效内存管理的示例代码

《Redis实现高效内存管理的示例代码》Redis内存管理是其核心功能之一,为了高效地利用内存,Redis采用了多种技术和策略,如优化的数据结构、内存分配策略、内存回收、数据压缩等,下面就来详细的介绍... 目录1. 内存分配策略jemalloc 的使用2. 数据压缩和编码ziplist示例代码3. 优化的

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象