当前位置: 首页 > news >正文

卡顿检测与 Choreographer 原理

一、卡顿检测的原理

卡顿的本质是主线程(UI 线程)未能及时完成某帧的渲染任务(超过 16.6ms,以 60Hz 屏幕为例),导致丢帧(Frame Drop)。检测卡顿的核心思路是监控主线程任务的执行时间

常见检测方法:
  1. 主线程监控(Looper Printer)

    • 原理:利用 LooperPrinter 机制,在每条消息处理前后打印日志,统计消息执行时间。
    • 代码示例:
      Looper.getMainLooper().setMessageLogging(printer -> {if (printer.startsWith(">>>>> Dispatching")) {startTime = System.currentTimeMillis();} else if (printer.startsWith("<<<<< Finished")) {long cost = System.currentTimeMillis() - startTime;if (cost > 16) reportBlock(cost);}
      });
      
  2. Choreographer FrameCallback

    • 原理:通过向 Choreographer 注册 FrameCallback,在每一帧开始绘制时触发回调,计算帧耗时。
    • 代码示例:
      Choreographer.getInstance().postFrameCallback(new FrameCallback() {@Overridepublic void doFrame(long frameTimeNanos) {long current = System.currentTimeMillis();if (lastFrameTime > 0) {long cost = current - lastFrameTime;if (cost > 16) reportJank(cost);}lastFrameTime = current;Choreographer.getInstance().postFrameCallback(this); // 持续监控}
      });
      
  3. Systrace/Perfetto

    • 原理:系统级工具,通过插桩代码和内核事件,分析每一帧的耗时及卡顿原因。
  4. BlockCanary 等开源库

    • 原理:基于主线程监控,结合堆栈采样,定位耗时方法。

二、Choreographer 的工作原理

Choreographer 是 Android 渲染系统的核心协调者,负责接收 VSync 信号并调度 UI 渲染流程。

1. 核心职责
  • 接收来自 SurfaceFlinger 的 VSync 信号(垂直同步信号,通常 60Hz)。
  • 按优先级调度以下任务:
    • INPUT(输入事件)
    • ANIMATION(属性动画)
    • TRAVERSAL(View 的 measure/layout/draw)
    • COMMIT(提交渲染结果)
2. 关键流程
  1. VSync 信号到达

    • 由硬件或软件模拟生成,通知应用开始新一帧的渲染。
  2. 任务调度(Choreographer)

    • Choreographer 根据 VSync 信号,依次触发注册的 FrameCallback
    • 调用 doFrame() 方法,依次处理输入、动画、视图遍历等任务。
  3. UI 渲染阶段

    • Measure/Layout/Draw:View 系统遍历视图树,生成绘制命令。
    • Sync & Upload:将 UI 数据同步到 RenderThread。
    • Draw:RenderThread 通过 OpenGL/Vulkan 将数据提交给 GPU。
  4. 提交到 SurfaceFlinger

    • 最终由 SurfaceFlinger 合成所有 Layer,输出到屏幕。
3. 代码流程
// Choreographer 核心逻辑
void doFrame(long frameTimeNanos, int frame) {// 1. 计算掉帧情况if (jitterNanos >= mFrameIntervalNanos) {Log.w(TAG, "Frame time jitter: " + jitterNanos);}// 2. 按顺序处理任务mFrameInfo.markInputHandlingStart();doCallbacks(Choreographer.CALLBACK_INPUT, frameTimeNanos);mFrameInfo.markAnimationsStart();doCallbacks(Choreographer.CALLBACK_ANIMATION, frameTimeNanos);mFrameInfo.markPerformTraversalsStart();doCallbacks(Choreographer.CALLBACK_TRAVERSAL, frameTimeNanos);doCallbacks(Choreographer.CALLBACK_COMMIT, frameTimeNanos);
}

三、卡顿与 Choreographer 的关系

  1. 卡顿的根本原因

    • 主线程在 doFrame() 中执行的任务(如 View 绘制、动画计算)耗时过长,导致未能在下一个 VSync 信号到来前完成渲染。
  2. Choreographer 的监控能力

    • 通过 postFrameCallback 可以精确测量两帧之间的时间差,判断是否发生丢帧。
    • 系统内部会统计 jankyFrames(掉帧数),并通过 onJankyFrames 回调通知应用。

四、优化卡顿的建议

  1. 减少主线程任务
    • 耗时操作(IO、网络、计算)移至子线程。
  2. 优化 View 层级
    • 避免过度绘制,减少 measure/layout/draw 耗时。
  3. 合理使用动画
    • 优先使用硬件加速的属性动画(如 ViewPropertyAnimator)。
  4. 监控工具结合
    • 使用 Systrace 分析渲染流水线,定位阻塞点。

总结

  • 卡顿检测依赖对主线程任务耗时的监控,可通过多种工具实现。
  • Choreographer 是 Android 渲染系统的中枢,通过 VSync 信号驱动 UI 渲染流程,其调度机制直接影响帧率稳定性。
  • 深入理解这两者的原理,是优化应用流畅性的关键基础。
http://www.xdnf.cn/news/478369.html

相关文章:

  • Java大师成长计划之第24天:Spring生态与微服务架构之分布式配置与API网关
  • window 显示驱动开发-使用有保证的协定 DMA 缓冲区模型
  • 论信息系统项目的范围管理
  • 后端框架(3):Spring(2)
  • Gitee DevOps:中国企业数字化转型的“本土化加速器“
  • Ubuntu 更改 Nginx 版本
  • PCIE接收端检测机制分析
  • 源码:处理文件格式和字符集的相关代码(3-3)
  • Qt图表绘制(QtCharts)- 性能优化(13)
  • 关于Redisson分布式锁的用法
  • TRTC实时对话式AI解决方案,助力人机语音交互极致体验
  • Python 编程技巧 @ 玩转 For 循环
  • Linux `ps` 命令深度解析与高阶应用指南
  • 简单介绍C++中线性代数运算库Eigen
  • 【未完】【GNN笔记】EvolveGCN:Evolving Graph Convolutional Networks for Dynamics Graphs
  • sqli-labs靶场29-31关(http参数污染)
  • ECPF 简介
  • python爬虫实战训练
  • vscode debug node + 前端
  • 学习51单片机02
  • Vue.js---计算属性computed和lazy
  • 简单图像自适应亮度对比度调整
  • 【Python-Day 14】玩转Python字典(上篇):从零开始学习创建、访问与操作
  • Flutter目录结构介绍、入口、Widget、Center组件、Text组件、MaterialApp组件、Scaffold组件
  • 【RK3588嵌入式图形编程】-Cairo-绘图基础-线条
  • Armijo rule
  • 从另一个视角理解TCP握手、挥手与可靠传输
  • k8s灰度发布
  • MES系统与ERP、SCM、QMS、APS系统的关系
  • 蓝牙网关都有哪些型号?