教你更好的使用硬件加速

原文:Hardware Acceleration
翻译:D.Winter

从Android 3.0 (API level 11)开始,Android 2D渲染管线能更好的支持硬件加速。硬件加速通过GPU执行各种绘画操作。因为硬件加速需要消耗更多的资源,所以你的App需要更多的RAM。
开启硬件加速最简单的方法是在整个应用全局设置。如果应用只使用标准的ViewDrawable,全局设置不会产生不利的影响。然而,因为硬件加速不支持所有的2D绘制操作,开启会影响一些自定义View或者绘制调用。问题显示为不可见的元素、异常,或者错误渲染像素。为了补救这些,Android提供给你选项开启或者关闭硬件加速在以下几个层面:
* Application
* Activity
* Window
* View

如果你的应用执行自定义绘制,开启硬件加速,测试应用在实际的硬件设备上去查找问题。

* 控制硬件加速*

你可以在以下层面控制硬件加速
* Application
* Activity
* Window
* View

Application 层

在你的Android manifest文件里,添加以下属性到标签里,针对整个应用开启硬件加速:

Activity 层

如果整个应用开启硬件加速表现的不稳定,你也可以针对单个Activity进行控制。在Activity层开启或者关闭硬件加速,你可以使用android:hardwareAccelerated属性在标签内。下面是单个Activity中关闭硬件加速的例子:




Window 层

如果你需要更细致的控制,可以在获得的Window里开启硬件加速:

getWindow().setFlags(
WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED,
WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED);


* 注: 在window层不能取消硬件加速。

View 层

通过下面代码,你可以在运行时关闭硬件加速:

myView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);


注:你不能在view层开启硬件加速。 View面板有其他方法关闭硬件加速。更多详细内容请看View layers|View面板。

* 判断View是否已经硬件加速*

有时候,知道一个应用当前是否硬件加速是非常有用的,尤其像自定义View。这非常有用,当你的应用做大量的自定义绘制并且不是所有操作都支持新的渲染管道。

有两个方法查看应用是否硬件加速:

* Android绘画模型*

当开启硬件加速,新的绘画模型利用显示列表在界面上渲染你的应用。
为了完全理解显示列表和对你的应用有怎样的影响,知道Android不通过硬件加速怎么样绘制View也很重要。下面的篇章阐述基于软件和基于硬件的绘画模型。

* 基于软件的绘画模型*

在软件绘画模型中,View通过以下两个步骤被绘制:

  • 使层次失效
  • 绘制层次

每当应用需要更新部分UI时,调用invalidate()(或者它的变形)在任意需要改变内容的view里。这些失效信息被传播始终在view层,计算需要重绘的界面区域。然后,Android系统绘制任意view在这些区域。不幸的是,这种模型有两个缺点:
* 第一,这种模型需要执行的大量的代码在每一次绘画。比如,如果你的应用在按钮里调用了invalidate() ,而且这个按钮在另外一个view之上,此时Android系统会重绘这个view,即时它没有发生改变。
* 第二个问题是绘画模型会隐藏你应用的错误。从Android系统开始重绘view,当他们与脏区域融合时,你改变的view的内容可能会被重绘即时invalidate()没有被调用。此时,你依赖于另外一个view使之失效来达到合适的反应。这个反应会在任何时候改变在你修改你的应用时。因此,你需要一直调用invalidate()在你的自定义view上,无论你修改数据或者影响你view绘画代码的声明。

注: Android view当它的属性改变时,自动调用invalidate() ,比如背景颜色或者文本框内容。

* 基于硬件的绘画模型*

Android仍使用invalidate()和draw()来响应界面更新和视图渲染,不同的是控制当前绘制。Android系统将他们记录在显示列表来代替马上执行绘画命令,它包含view层绘画代码的输出。另外一个优化是Android系统只需对记录和更新显示列表,通过调用invalidate()将view标记为。
还没失效的View通过重新运行之前的记录显示列表被重绘。这个新绘画模型包括三个阶段:

  • 1、层失效
  • 2、记录更新显示列表
  • 3、绘画显示列表

用此模式,你不能依赖与脏区域交叉的view来执行它的draw() 。为了确保Android系统记录一个View的显示列表,你必须调用invalidate() 。
忘记这么做会导致视图看起来一样,甚至在改变它后。一旦发生,这个BUG很容易被发现。

使用显示列表也利于动画增强,因为设置特殊的属性,像透明度和旋转, 不需要使目标视图失效(它自动完成)。这个优化也适用于显示列表的视图 (你应用的任一视图都硬件加速) 比如,假设有个LinearLayoutButton上有个ListView。LinearLayout的显示列表看上去这样:

DrawDisplayList(ListView)
DrawDisplayList(Button)
假设你现在要改变ListView的不透明,在调用setAlpha(0.5f)后,显示列表变成:
SaveLayerAlpha(0.5)
DrawDisplayList(ListView)
Restore
DrawDisplayList(Button)
ListView的设置代码没有被执行。系统只更新显示列表中更简单的LinearLayout。

在未开启硬件加速的应用中,列表的绘制代码在其父亲中还会被执行一次。

* 不支持的绘画操作*

当启动硬件加速, 2D渲染通道支持一般使用的画布绘画操作和一些较少使用的操作。所有的绘画操作被用于渲染程序,默认为widget和layout,还有些高级应用比如反光和纹理平铺也是被支持的。下面是不被硬件加速的操作清单:

另外, 有些操作在硬件加速开启后会发生变化:

如果你的应用受缺失属性和限制的影响,你可以关闭硬件加速,在你受影响的部分调用[android.graphics.Paint) setLayerType(View.LAYER_TYPE_SOFTWARE, null)](http://docs.eoeandroid.com/reference/android/view/View.html#setLayerTypeint,)( 。这个方法, 你仍可以利用硬件加速其他任一地方。查看Controlling Hardware Acceleration得到关于硬件加速的更多信息。

* View 层*

在Android的不同版本, view已经有能力渲染进入屏幕缓存区内,无论是view的绘制缓存,还是使用[android.graphics.Paint, int) Canvas.saveLayer()](http://docs.eoeandroid.com/reference/android/graphics/Canvas.html#saveLayerandroid.graphics.RectF,)。屏幕缓存,或层,有多种用途。你可以使用他们获得更好的性能,当动画组合视图或者需要应用复合效应。比如,你可以使用Canvas.saveLayer)来实现消退效果来临时渲染一个view进入层,然后使用opacity( factor合成到界面。

Android 3.0 (API level 11)开始,你有更多的控制通过[android.graphics.Paint) View.setLayerType()](http://docs.eoeandroid.com/reference/android/view/View.html#setLayerTypeint,)方法,怎样或者什么时候使用layers。这个API有2个参数:layer的类型和可选的[Paint]http://docs.eoeandroid.com/reference/android/graphics/Paint.html)对象,它阐明layer怎么样合成。(
你可以使用Paint参数使用滤色镜,特殊的混合模式,或者设置为不透明。view可以使用三种layer类型的其中一种:

The type of layer you use depends on your goal:

    • Performance* : 由硬件渲染到硬件纹理,一旦View被渲染到layer,它的绘图代码不会被执行直到调用invalidate() 。有些动画,像透明度动画,直接放入layer,由GPU完成非常有效率。
    • Visual effects* :使用硬件或者软件layer类型和Paint,对view应用特殊的视觉处理。比如,你使用ColorMatrixColorFilter绘制一个view为黑色或者白色。
    • Compatibility* :使用软件layer类型促使view软件渲染。如果硬件加速的,有着渲染问题, 这是一个简单的方法来绕过限制的硬件渲染管道。

* View层和动画*

开启硬件加速,硬件层提供提供更快的和更平滑的动画效果。当有很多绘图操作时,动画每秒60帧不是一直能保持的。硬件层可以减轻这个,通过渲染为硬件纹理。硬件纹理可以优化View,不再需要视图不断重绘本身。当你调用invalidate()或者改变view属性时,view才会重绘。如果动画显示的不够平滑,考虑开启硬件层在你使用的View。

当view进入后台硬件层,层被混合到界面,view的属性被控制。设置这些属性很有效果,因为他们不需要view失效或者重绘。以下属性作用于混合层。
通过setter测试属性,获得最优效果:

  • alpha: 改变层透明度
  • x, y, translationX, translationY: 改变层位置
  • scaleX, scaleY: 改变层大小
  • rotation, rotationX, rotationY: 改变层的三维定位
  • pivotX, pivotY: 改变层的转换源 当view被作为ObjectAnimator启用时,这些属性被使用其他名字。如果你想使用这些属性,请调用适当的setter或者getter。比如,为了改变alpha属性,调用setAlpha() 。下面的代码片段展示了最有效的方法旋转viewiew在3D的Y轴上:

view.setLayerType(View.LAYER_TYPE_HARDWARE, null);
ObjectAnimator.ofFloat(view, "rotationY", 180).start();

因为硬件层消耗媒体资源,强烈介意你只在持续动画时开启,并且在动画结束时关闭。你可以使用animation listeners完成:

View.setLayerType(View.LAYER_TYPE_HARDWARE, null);
ObjectAnimator animator = ObjectAnimator.ofFloat(view, "rotationY", 180);
animator.addListener(new AnimatorListenerAdapter() {
@Override
public void onAnimationEnd(Animator animation) {
view.setLayerType(View.LAYER_TYPE_NONE, null);
}
});
animator.start();

更多信息,请看Property Animation

* 要领和技巧*

开启硬件加速2D图像可以立即提高性能, 但你仍可以设计你的应用使用GPU性能,以下推荐:

  • 减少应用中视图的数量* 视图越多,速度越慢. 请求软件渲染通道也越慢。减少视图是优化UI的最简单的办法。
  • 避免透支* 不要在顶层绘制太多。去除那些被完全遮挡住的图层。如果你要绘制相互叠加的多个图层,考虑将他们合并为一个图层。
  • 不要在draw方法中创建render对象* 一个通常的错误是创一个Paint对象或者Path对象,在任何时候rendering方法被调用。这使得垃圾收集器运行的更频繁,忽视缓存,在硬件通道优化。
  • 不要太频繁的改变图形* 图形实例化时,由texture masks被渲染。任何时候你改变路径, 硬件通道创建一个新的mask,这是非常耗资源的。
  • 不要太频繁的改变位图* 任何时候你改变一个位图的内容,它作为GPU纹理重新被上传在下一次绘制时。
  • 使用透明度要小心* 当你通过setAlpha()来改变透明度时, AlphaAnimation,或者ObjectAnimator,在off-screen buffer被渲染,它两倍于使用fill-rate。当应用alpha在非常大的界面上时,考虑设置视图属性LAYER_TYPE_HARDWARE.
  • 转自:http://wiki.eoe.cn/page/Hardware_Acceleration.html

转载于:https://my.oschina.net/u/150685/blog/281837

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值