JellyBean的VSync与三级buffer的入门介绍(外文翻译)

2023-10-11 01:32

本文主要是介绍JellyBean的VSync与三级buffer的入门介绍(外文翻译),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近在研究JellyBean在Graphics上面的新改动,最吸引人的在图像的平滑上大作文章,读了一篇关于VSync和三级缓冲的外国文章,很有收获,特别翻译成中文,与大家分享。

原文地址:http://www.androidpolice.com/2012/07/12/getting-to-know-android-4-1-part-3-project-butter-how-it-works-and-what-it-added/

https://www.codeaurora.org/gitweb/quic/la/

今天我们来探讨一些不同的,来研究一些对于所有人都很实用的一个话题:性能。大家可以看到JellyBean非常的快,在GalaxyNexus测试它可以感到它变成了一台全新的手机。滚动变得非常平滑和更快了,而且相应点击也变得高度敏感。另外,这些平滑的表现体现在了各个方面。

           我不清楚你们是否都已经看到了这些改变,不过这些都很无趣呀,真正有趣的东西是——他们怎么做到这些的?这就是我们这里要探究的。因此,抓起你的爆米花,孩子;是时候开始学习“黄油计划”(ProjectButter)了。


他们是怎么做的

那么你们是怎么让一个八个月大的GalaxyNexus跑起来像是Galaxy SIII一样?答案是,许许多多努力的结果。在GoogleI/O大会上我们已经看到了,由我最喜欢两个I/O演讲者,ChetHaaseRomain Guy。一个小时长的PPT展示对于开发者似乎长了一点,所以让我从中挑选一些有意思的事。


VSync使得帧数被放入了一个充满润滑油的机器里



PC游戏者们可能对于“VSync”这个词比较熟悉,它是一个图像选项可以防止你的屏幕发生图像不同步(imagetearing)的情况。为了了解什么事VSync,我们来补充一点背景知识:影像(比如手机显示的那样)是通过一张一张的图,叫做帧,构成的。平滑的动画通常是一秒60帧。帧是由像素构成,而且,当显示的时候,像素是一行一行的打上去的。明白了么?很好。

显示器(LED,AMOLED,或其他)从图形芯片上面获得每一帧,然后开始一行一行的绘制。理想情况下,你希望在下一帧还未出现时就已经被绘制好了,不同步或者叫图像撕裂(tearing)的情况出现在当芯片还未完全生成新的一帧,LCD已经开始显示了,这样你就会看到一半新的一半老的帧。

  VSync,同步了这样一个过程,它告诉GPU去等屏幕显示完上一帧再去加载下一帧。

Android一直都是用的VSync来防止屏幕发生不同步,但是JellyBean把这件事做的更好。现在VSync被用在了所有的进程上用来显示下一帧。

            大多数android显示是保持在每秒60帧的水平,即60Hz。为了有一个平滑的动画效果,为了能够保持这种60Hz的状态不变,这意味着你只有16ms的时间来处理每一帧。如果你超过16ms,动画会发生迟滞,那种犹如黄油般的感觉就会荡然无存。

            16毫秒的时间不算太长,所以你要充分的利用它。在IceCreamSandwich中,对下一帧的展现可以说算得上是一种“懒加载”的形式,在JellyBean里,在VSync脉冲开始时,所有的进程会尽可能快的在上一帧完成时显示下一帧。换句话说,他们是尽可能的充分利用那16ms。如下是一个例子



             这就是没有加入VSync的情况。在这张图里,数字代表的是帧,帧是由CPUGPU处理的,当它被处理完之后会被放在下一个VSync脉冲周期里去显示。所以在这张图片里,0号帧被显示了,在16ms的显示时间内,CPUGPU准备好了下一帧,计算如期完成,在下一个VSync脉冲周期里,1号帧很好的显示出来了,很好!

             现在我们正在显示1号帧,开始处理2号帧,有一些东西降低了系统的性能,2号帧在这个脉冲周期结束临近时还没有处理完成。这时候只有4ms来处理2号帧,所以它没有被及时处理完,没有2号帧,显示只能继续显示1号帧,并且又是一个16msAndroid团队称之为为“Jank”,他的大意是动画在这个时间点上变得不流畅,用户会有迟滞的感觉。


             这里是JellyBean如何做的,所有帧的处理都被放在了每一个VSync脉冲周期的最开始,这样一来,所有的16ms都被充分的利用了。帧的处理从最开始的“好吧,被我碰到了我就来处理”的模式到现在的刚性时间规划,高度的有组织。在这个例子里,你会获得一个黄油一般平滑的体验。


三级缓冲让Jank的几率更加低


             VSync不是唯一提高动画平滑度的事情,Android还能提供一种从性能下降之后恢复平滑的办法。


             那么到底什么又是“buffer”?简单来说,buffer就是帧在被创建和储存的一个容器。在前面,我们将帧标上了编号,但事实上,这些帧都是放置在两个buffer中的。Android是一个双缓冲的,很典型的架构,这意味着它可以显示着一个并且处理着另外一个。在图中我们可以看到buffer被标上了“A”和“B”。当显示着A的时候,B正在被处理中。当A显示完成,B也准备完成之后,他们交换角色,B被显示,A被清理然后开始处理新的一帧。


             当某一帧的处理时间超过16ms时,双buffer的问题就暴露了出来。在图里我们看到当bufferB的处理时间超过了16ms,就意味着迟滞(Jank)要发生了。迟滞是一件坏事,但是更要命的是,图中的CPUGPU的空白部分都是浪费掉的时间片。在显示第一帧时,bufferB时间超了,因此bufferB会被占用着直到它被显示,bufferA也被占用着,因为他要继续显示直到B可以显示为止。而且要记住一点,buffer的切换只能发生在VSync脉冲周期开始的时候,CPUGPU没有了可以使用的buffer,因此他们只能等在哪里。一次的减速会带来后续的速度降低,这就是4.0如何工作的。



              在JellyBean中,我们有了关于这种问题的解决办法,第三个buffer!一样的情况像刚才一样,bufferB使用了很长的时间,A被用作显示在这时候,与其让CPU,GPU等在哪里,系统创建了bufferC,来处理下一帧。三级缓冲停止了这种迟滞的蔓延。当跳过了一个初始的迟钝之后,用户在后面看到将是一个平滑的动画,这都是因为提供给显示一个临时的跳板,这样即使在程序下面并不是运行的那么顺利,用户所看到的依然很流畅。

              所以为什么他们不是一直使用三级缓冲的技术呢?那么,我们再从图中看一下,三级缓冲引入了一些点击、输入的延时,当bufferC开始运行到显示的这段时间,你可能会受到两种感受,一种是输入的延时,另一种是起伏的动画(这里作者用的是choppy animation,不知道该怎么理解)。

              鉴于此,JellyBean并没有任何时候都使用三级缓冲,它通常是使用双buffer,但是三级缓冲会在你需要的时候出现。这样做也是为了减少输入的延时,当有问题出现的时候,三级缓冲才会跳出来帮助你恢复画面迟滞的现象。

这篇关于JellyBean的VSync与三级buffer的入门介绍(外文翻译)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中HashMap的用法详细介绍

《Java中HashMap的用法详细介绍》JavaHashMap是一种高效的数据结构,用于存储键值对,它是基于哈希表实现的,提供快速的插入、删除和查找操作,:本文主要介绍Java中HashMap... 目录一.HashMap1.基本概念2.底层数据结构:3.HashCode和equals方法为什么重写Has

Springboot项目构建时各种依赖详细介绍与依赖关系说明详解

《Springboot项目构建时各种依赖详细介绍与依赖关系说明详解》SpringBoot通过spring-boot-dependencies统一依赖版本管理,spring-boot-starter-w... 目录一、spring-boot-dependencies1.简介2. 内容概览3.核心内容结构4.

Java List 使用举例(从入门到精通)

《JavaList使用举例(从入门到精通)》本文系统讲解JavaList,涵盖基础概念、核心特性、常用实现(如ArrayList、LinkedList)及性能对比,介绍创建、操作、遍历方法,结合实... 目录一、List 基础概念1.1 什么是 List?1.2 List 的核心特性1.3 List 家族成

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

史上最全MybatisPlus从入门到精通

《史上最全MybatisPlus从入门到精通》MyBatis-Plus是MyBatis增强工具,简化开发并提升效率,支持自动映射表名/字段与实体类,提供条件构造器、多种查询方式(等值/范围/模糊/分页... 目录1.简介2.基础篇2.1.通用mapper接口操作2.2.通用service接口操作3.进阶篇3

Python自定义异常的全面指南(入门到实践)

《Python自定义异常的全面指南(入门到实践)》想象你正在开发一个银行系统,用户转账时余额不足,如果直接抛出ValueError,调用方很难区分是金额格式错误还是余额不足,这正是Python自定义异... 目录引言:为什么需要自定义异常一、异常基础:先搞懂python的异常体系1.1 异常是什么?1.2

Python实现Word转PDF全攻略(从入门到实战)

《Python实现Word转PDF全攻略(从入门到实战)》在数字化办公场景中,Word文档的跨平台兼容性始终是个难题,而PDF格式凭借所见即所得的特性,已成为文档分发和归档的标准格式,下面小编就来和大... 目录一、为什么需要python处理Word转PDF?二、主流转换方案对比三、五套实战方案详解方案1:

setsid 命令工作原理和使用案例介绍

《setsid命令工作原理和使用案例介绍》setsid命令在Linux中创建独立会话,使进程脱离终端运行,适用于守护进程和后台任务,通过重定向输出和确保权限,可有效管理长时间运行的进程,本文给大家介... 目录setsid 命令介绍和使用案例基本介绍基本语法主要特点命令参数使用案例1. 在后台运行命令2.

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

Spring WebClient从入门到精通

《SpringWebClient从入门到精通》本文详解SpringWebClient非阻塞响应式特性及优势,涵盖核心API、实战应用与性能优化,对比RestTemplate,为微服务通信提供高效解决... 目录一、WebClient 概述1.1 为什么选择 WebClient?1.2 WebClient 与