让Android TextureView Canvas支持硬件加速

SurfaceViewTextureView是Android提供的支持异步线程渲染的控件,虽然都是异步渲染,但不论在使用方式上还是在底层实现原理上都是有差异的。

SurfaceView在内部是直接创建一个独立于当前Activity的Surface,此Surface默认处于Activity Surface的下面,然后在Activity上对应位置挖一个相应大小的透明窟窿,使下面的SurfaceView的内容能够展示出来。

TextureView 的内容是正常渲染到当前Activity的Surface上的。其内部是基于SurfaceTexture实现的,当渲染侧渲染完成后,SurfaceTexture会将渲染内容以纹理的形式上传到GPU,接着由TextureView将纹理渲染到当前Activity的视图树中。

在使用上,SurfaceViewTextureView 都提供了Canvas接口给应用侧进行异步渲染,但SurfaceView 可以由应用指定Canvas是否走硬件加速流程,而TextureView 提供的TextureView.lockCanvas()TextureView.lockCanvas(Rect dirty)接口返回的Canvas都是非硬件加速的,因此渲染指令的执行完全是由CPU执行的,虽然是异步线程,但是在渲染复杂内容时,一方面单帧渲染耗时高导致渲染帧率低,另一方面会消耗大量CPU资源,导致应用整体卡顿以及设备发热、掉电等问题。

解决方案

使用liuwons/HwTextureView

使用方式和普通TextureView 接本一致,lockCanvas()获取的Canvas在Android 28以上平台具有硬件加速能力。需要注意的是lockCanvas(Rect dirty)方法不可用(直接返回null)。

内部实现是创建了一个Surface,作为图形生产者,消费者侧连接了TextureViewSurfaceTexture,使用此Surface对外提供硬件加速接口。

渲染性能对比

测试工程: TextureViewPlayground - Github

测试设备: Google Pixel 7

绘制内容: 每一帧绘制相同的30笔不透明/半透明的stroke path 绘制结果:

统计内容:

  • 每一帧的渲染时长(只统计所有drawPath方法的执行总耗时,不包含lockCanvas和unlockCanvasAndPost方法执行耗时):
  • CPU使用率

统计结果:

TextureView HwTextureView
渲染时长 10ms 1ms
CPU 13ms 8ms

TextureView测试数据

HwTextureView测试数据

相关推荐
NiceCloud喜云4 小时前
Opus 4.8 的 Effort Control 怎么选:Low 到 Max 五档策略
android·java·大数据·前端·c++·python·spring
日光明媚8 小时前
一步生成视频!One-Forcing:DMD + 零成本 GAN,训练 200 步超越多步 SOTA
android·开发语言·kotlin
帅次9 小时前
Android 17 开发者实战:核心更新与应用场景落地指南
android·java·ios·android studio·iphone·android jetpack·webview
大鹏说大话9 小时前
SQL 排序与分组实战:解决“分组后取最新数据“
android·java·数据库
搜狐技术产品小编202312 小时前
破局与重构:纯端侧 Android 自动化引擎的尝试与未来推演
android·运维·重构·自动化
码云骑士12 小时前
Android SystemServer启动过程
android·systemserver
weiggle13 小时前
第三篇:可组合函数(Composable)——Compose 的基石
android·前端
独隅14 小时前
Android Studio 接入多种不同 AI 大模型进行开发的全面详细指南(Android Studio+AI)
android·人工智能·android studio
夜微凉414 小时前
三、MySQL
android·数据库·mysql
我命由我1234514 小时前
Android 开发问题:项目同时引入了两个包含相同类文件的库(AndroidX 库、旧版本支持库),导致了重复类错误
android·java·java-ee·android studio·android-studio·androidx·android runtime