【学习笔记】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类的相关方法实现,本文之所以这样写,本是想着每绘制完一帧后,开始处理下一帧,看哪种方法更快。

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/24645.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringBoot高手之路-springboot原理篇

配置文件优先级 SpringBoot原理篇-多环境配置

[职场] 社保和商业保险有什么区别?可以只买商保不买社保吗? #微信#经验分享#媒体

社保和商业保险有什么区别&#xff1f;可以只买商保不买社保吗&#xff1f; 我们在提到社保和商业保险时&#xff0c;经常会听到这样一句话&#xff1a;“社保是基础&#xff0c;商保是补充。” 为什么会这样说呢&#xff1f;社保和商保有什么区别呢&#xff1f;今天&#xf…

Springboot 通过SSE 实现实时消息返回

网上搜了好多都是用 SseEmitter 实现的,自己搭的demo确实也可以了,但是我项目里有一个过滤器,死活配置都不行,终于用google搜了一下,第一篇帖子便解决了这个问题,代码和大佬链接如下: https://github.com/CodingChaozhang/spring_boot_practice_demo/blob/master/springboot_s…

XR和Steam VR项目合并问题

最近有一个项目是用Steam VR开发的&#xff0c;里面部分场景是用VRTK框架做的&#xff0c;还有一部分是用SteamVR SDK自带的Player预制直接开发的。 这样本身没有问题&#xff0c;因为最终都是通过SteamVR SDK处理的&#xff0c;VRTK也管理好了SteamVR的逻辑&#xff0c;并且支…

数据结构之初识泛型

目录&#xff1a; 一.什么是泛型 二.引出泛型 三.泛型语法及&#xff0c;泛型类的使用和裸类型(Raw Type) 的了解 . 四.泛型的编译&#xff1a; 五.泛型的上界 六.泛型方法 注意&#xff1a;在看泛型之前可以&#xff0c;回顾一下&#xff0c;包装类&#xff0c;包装类就是服务…

JSP中连接数据库MySQL

JSP中连接数据库MySQL 一、软件环境 下载并安装MySQL&#xff0c;Tomacat&#xff0c;JDBC、IDEA或其他IDE&#xff0c;本文使用IDEA 二、环境配置 将其环境变量配置好之后&#xff0c;下载Java 专用的连接MySQL的驱动包JDBC。 下载链接&#xff1a;https://dev.mysql.com/…

AI高考大战,揭秘五大热门模型谁能问鼎数学之巅?

在高考前&#xff0c;我就有想法了&#xff0c;这一次让AI来做做高考题。就用国内的大模型&#xff0c;看哪家的大模型解题最厉害。 第一天考完&#xff0c;就拿到了2024高考数学2卷的电子版&#xff0c;这也是重庆市采用的高考试卷 这次选了5个AI工具&#xff0c;分别是天工&a…

帕友饮食改善的小建议!

一、增加膳食纤维的摄入 帕金森病患者应增加膳食纤维的摄入量&#xff0c;以帮助调节肠道功能&#xff0c;预防便秘。膳食纤维丰富的食物包括蔬菜、水果、全谷类食物等。患者应确保每天摄入足够的膳食纤维&#xff0c;以保持肠道通畅&#xff0c;缓解帕金森病可能带来的消化不…

[AIGC] Springboot 自动配置的作用及理由

在详细解释SpringBoot的自动配置之前&#xff0c;先介绍以下背景知识。在创建现代复杂的应用程序时&#xff0c;一个困难的部分是正确地设置您的开发环境。这个问题尤其在Java世界中尤为突出&#xff0c;因为您必须管理和配置许多独立的标准和技术。 当我们谈论Spring Boot的自…

【Unity】官方文档学习-光照系统

目录 1 前言 2 光照介绍 2.1 直接光与间接光 2.2 实时光照与烘焙光照 2.3 全局光照 3 光源 3.1 Directional Light 3.1.1 Color 3.1.2 Mode 3.1.3 Intensity 3.1.4 Indirect Multiplier 3.1.5 Shadow Type 3.1.6 Baked Shadow Angle 3.1.7 Realtime Shadows 3.1…

数据挖掘--分类

数据挖掘--引论 数据挖掘--认识数据 数据挖掘--数据预处理 数据挖掘--数据仓库与联机分析处理 数据挖掘--挖掘频繁模式、关联和相关性&#xff1a;基本概念和方法 数据挖掘--分类 数据挖掘--聚类分析&#xff1a;基本概念和方法 基本概念 决策树归纳 决策树:决策树是一…

如何提高网站访问量?

提高网站访问量通常需要一个多方面的策略&#xff0c;涉及SEO、内容营销、社交媒体和其他网络营销手段&#xff0c;而我们仅从seo入手来说说 关键词优化是SEO的核心&#xff0c;它确保网站能够针对潜在用户的搜索查询进行优化。这不仅涉及在网站内容中使用正确的关键词 还需要…

问题:前肢的前方称() #微信#经验分享#微信

问题&#xff1a;前肢的前方称&#xff08;&#xff09; A . 掌侧 B . 跖侧 C . 背侧 D . 胫侧 E . 桡侧 参考答案如图所示

TOGAF数字化转型的关键(文尾附在线TOGAF免费测试)

业务架构驱动数据架构和应用架构的设计&#xff0c;而应用架构又依赖于数据架构和技术架构的支持。技术架构则为整个架构提供了稳定的基础设施。 在数字化转型中&#xff0c;协调和整合这四种架构是至关重要的。通过确保它们之间的一致性和协同工作&#xff0c;可以实现企业业务…

极简主义在UI设计中的应用及解析

极简主义&#xff0c;即“少就是多”。在设计中&#xff0c;极简主义是许多艺术概念之一&#xff0c;它描述了一种内容形式&#xff0c;可以在许多方面使用。现在移动UI界面和网页设计中的极简主义设计越来越多。即时设计认为&#xff0c;极简主义UI界面不仅美观&#xff0c;而…

[经验] 昆山教育网(昆山教育网中小学报名) #媒体#职场发展#微信

昆山教育网&#xff08;昆山教育网中小学报名&#xff09; 昆山教育局网站 网站&#xff1a;昆山市教育局 昆山市教育局全面贯彻执行党和国家的教育方针、政策&#xff0c;落实有关教育工作的法律、法规&#xff1b;负责制定本市教育工作的实施意见和措施&#xff0c;并监督…

TriForce: 突破长序列生成瓶颈的分层投机解码技术

在人工智能领域&#xff0c;大型语言模型&#xff08;LLMs&#xff09;的长序列生成能力一直是研究的热点。然而&#xff0c;随着模型规模的增长&#xff0c;推理过程中的内存和计算瓶颈成为了限制其应用的主要障碍。为了解决这一问题&#xff0c;Carnegie Mellon University和…

1867java银证转账系统系统Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 java银证转账系统系统是一套完善的web设计系统&#xff0c;对理解JSP java编程开发语言有帮助采用了java设计&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统采用web模式&#xff0c;系统主要采用B/S模式开发。开发环境为TOMCAT7.0,Myeclipse8.5开发&a…

go语言后端开发学习(一)——JWT的介绍以及基于JWT实现登录验证

什么是JWT JWT,全名为JSON Web Token&#xff0c;是当下主流的一种服务端通信认证方式&#xff0c;具有轻量,无状态的特点&#xff0c;它实现了让我们在用户与服务器之间传递安全可靠的Json文本信息&#xff0c;它的使用过程主要是这样的&#xff1a; 当用户注册的时候&#x…

【百万字详解Redis】集群

文章目录 一、集群模式概述1.1、什么是集群模式1.2、集群模式特点1.3、集群工作方式 二、集群模式的搭建2.1、搭建前的准备2.2、修改集群配置2.3、启动redis服务2.4、创建集群2.5、查看redis服务状态2.6、进入一个节点2.7、测试操作 三、集群操作3.1、主从切换3.2、从节点操作3…