原文:Hardware Acceleration
翻译:D.Winter
从Android 3.0 (API level 11)开始,Android 2D渲染管线能更好的支持硬件加速。硬件加速通过GPU执行各种绘画操作。因为硬件加速需要消耗更多的资源,所以你的App需要更多的RAM。
开启硬件加速最简单的方法是在整个应用全局设置。如果应用只使用标准的View和Drawable,全局设置不会产生不利的影响。然而,因为硬件加速不支持所有的2D绘制操作,开启会影响一些自定义View或者绘制调用。问题显示为不可见的元素、异常,或者错误渲染像素。为了补救这些,Android提供给你选项开启或者关闭硬件加速在以下几个层面:
* Application
* Activity
* Window
* View
如果你的应用执行自定义绘制,开启硬件加速,测试应用在实际的硬件设备上去查找问题。
你可以在以下层面控制硬件加速
* Application
* Activity
* Window
* View
在你的Android manifest文件里,添加以下属性到标签里,针对整个应用开启硬件加速:
如果整个应用开启硬件加速表现的不稳定,你也可以针对单个Activity进行控制。在Activity层开启或者关闭硬件加速,你可以使用android:hardwareAccelerated属性在标签内。下面是单个Activity中关闭硬件加速的例子:
如果你需要更细致的控制,可以在获得的Window里开启硬件加速:
getWindow().setFlags(
WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED,
WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED);
* 注: 在window层不能取消硬件加速。
通过下面代码,你可以在运行时关闭硬件加速:
myView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
注:你不能在view层开启硬件加速。 View面板有其他方法关闭硬件加速。更多详细内容请看View layers|View面板。
有时候,知道一个应用当前是否硬件加速是非常有用的,尤其像自定义View。这非常有用,当你的应用做大量的自定义绘制并且不是所有操作都支持新的渲染管道。
有两个方法查看应用是否硬件加速:
当开启硬件加速,新的绘画模型利用显示列表在界面上渲染你的应用。
为了完全理解显示列表和对你的应用有怎样的影响,知道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通过重新运行之前的记录显示列表被重绘。这个新绘画模型包括三个阶段:
用此模式,你不能依赖与脏区域交叉的view来执行它的draw()。为了确保Android系统记录一个View的显示列表,你必须调用invalidate()。
忘记这么做会导致视图看起来一样,甚至在改变它后。一旦发生,这个BUG很容易被发现。
使用显示列表也利于动画增强,因为设置特殊的属性,像透明度和旋转, 不需要使目标视图失效(它自动完成)。这个优化也适用于显示列表的视图 (你应用的任一视图都硬件加速) 比如,假设有个LinearLayout在Button上有个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得到关于硬件加速的更多信息。
在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类型的其中一种:
LAYER_TYPE_NONE:被一般渲染并且不会被进入屏幕缓存。这是默认行为。
LAYER_TYPE_HARDWARE:由硬件渲染到硬件纹理,如果应用开启硬件加速。如果未开启, 就同LAYER_TYPE_SOFTWARE。
LAYER_TYPE_SOFTWARE:由软件渲染到位图。
The type of layer you use depends on your goal:
开启硬件加速,硬件层提供提供更快的和更平滑的动画效果。当有很多绘图操作时,动画每秒60帧不是一直能保持的。硬件层可以减轻这个,通过渲染为硬件纹理。硬件纹理可以优化View,不再需要视图不断重绘本身。当你调用invalidate()或者改变view属性时,view才会重绘。如果动画显示的不够平滑,考虑开启硬件层在你使用的View。
当view进入后台硬件层,层被混合到界面,view的属性被控制。设置这些属性很有效果,因为他们不需要view失效或者重绘。以下属性作用于混合层。
通过setter测试属性,获得最优效果:
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性能,以下推荐: