UI图像显示原理 & 卡顿掉帧

一.图像显示原理

关于CPU和GPU都是通过总线连接起来的
在CPU当中往往是一个位图,再经由总线在合适的时机传输给GPU, 然后GPU会把位图进行图层的渲染,包括文理的合成; 之后把结果放到Frame Buffer,由视频控制器根据Vsync信号,在制定的时间之前去提取FrameBuffer中对应屏幕显示内容,最终显示到手机屏幕上


CPU全部工作

  • Layout(布局):UI布局, 文字size计算等
  • Display(绘制):如drawRect
  • Prepare(准备)如图片编解码
  • Commint(提交):提交位图


  • 首先,我们创建一个UIView空间,显示部分交由CALayer负责. CALayer中有一个contents属性, 就是我们最终绘制到屏幕上的一个位图, 比如helloworld这个Label, contents里面是一个helloworld文字位图
  • 然后系统会在合适的时机,回调drawrect方法, 我们在此之上绘制自定义的内容, 绘制好的内容交由CoreAnimation这个框架提交OpenGL渲染管线, 进行最终的绘图渲染和文理合成, 然后显示到屏幕上

GPU渲染管线

  1. 顶点着色
  2. 图元装配
  3. 光栅化
  4. 片段着色
  5. 片段处理

二. UI卡顿和掉帧

知识背景

  • 页面滑动的流畅性是60fps, 即1s有60帧的画面更新才可以让眼睛感觉画面流畅
  • 在规定的事件内, 即16.67ms(按照一分钟60帧计算), 由CPU+GPU协同产生一帧的数据
  • CPU在做 UI布局, 文本计算, 绘制, 图片编解码等工作占用时间过长, 导致留给GPU的时间非常少, GPU想要把图层合成和文理渲染准备完毕, 所需要的总时间可能超过16.67ms,
  • 这样在下一帧VSync信号到来之前, 我们没有准备好下一帧画面, 就产生了掉帧现象, 我们肉眼看到的效果就是卡顿现象.


三 . 滑动方案优化

  • 方案一

CPU把下面操作放到子线程:

  1. 对象的创建, 调整, 销毁
  2. 预排班(布局计算, 文本计算)
  3. 预渲染(文本等异步绘制, 图片编解码等)
  • 方案二
GPU优化
  1. 文理渲染
  • 1.1 避免离屏渲染
  • 1.2 依托于CPU异步绘制机制减轻GPU压力
  1. 视图合成
  • 2.1 如果多个View层层叠加, GPU就要做每个视图的合成, 合成每一个像素点的像素值, 需要大量的计算
  • 2.2如果减轻视图的复杂性, 就可以减轻GPU的压力, 也可以采用CPU异步绘制, 使得提交的位图本身就是一个层级很少的视图, 也可以减轻GPU的压力

四. UI绘制原理&异步绘制

  • UIView调用setNeedsDislay(实际上是这个View的layer调用setNeedsDislay方法, 之后在layer上打上一个脏标记), 然后并没有立即发生当前视图的绘制工作, 而是在当前runloop快要结束的时候调用CALayer的display方法, 进入到当前视图的真正绘制工作中.
  • 原因要减少绘制次数, 提升性能, 所以要在当前runloop快要结束时调用CALayer的display方法

1. 系统绘制流程

  • 在CALayer内部会创建一个backing Store (我们可以理解为是CGContextRef)
  • 然后Layer会判断是否有代理
  • 没有代理的话会调用CALayer的drawInContext:方法
    有代理的话会调用代理的drawInContext方法, 然后做当前视图的绘制工作(发生在系统当中的). 然后在一个合适的时机回调一个系统的方法[UIView drawRect], drawRect的实现默认是什么都不做的, 而给我们开这个口子就是允许我们在系统绘制的基础之上绘制一些其他的工作
  • 最终无论上面哪一个分支, 都是由CALayer上传位图(backing store)给GPU
2. 异步绘制

问: 如何实现异步绘制?
答: 只要layer.delegate实现了displayLayer:方法, 我么就可以进行异步绘制

  • 代理负责生成对应的bitmap(位图)
  • 设置该bitmap(位图)作为layer.contents属性的值
异步绘制原理

通过子线程切换, 借助Global queue, 在子线程进行位图的绘制, 此时主线程可以做其他工作; 等子线程绘制完毕, 再回到主线程提交位图, 设置CAlayer的contents属性, 完成UI异步绘制过程.

五 . 离屏渲染

  1. 在屏渲染
  • 指的是GPU的渲染操作是在当前用于显示的屏幕缓冲区进行.
  1. 离屏渲染
    指的是GPU在当前屏幕缓冲区以外新开辟一个缓冲区进行渲染操作.
    也就是当我们设置某些UI视图层属性, 如果指令为:在未被合成前不能直接显示的时候, 典型的比如设置:
  • 圆角属性&&maskTobounds = YES(仅有一条是不会触发的),
  • 图层蒙版,
  • 阴影,
  • 光栅化,
    就会触发离屏渲染
  1. 为什么要避免离屏渲染
  • 离屏渲染是发生在GPU层面, 使得GPU触发了OpenGL的多通道渲染管线, 产生了额外的开销, 增加了GPU工作量, 可能使得CPU+GPU的工作时间超出了16.7ms的总耗时, 可能会导致UI卡顿和掉帧

总结

  1. 系统的UI传递机制是怎样的?
  1. 使UITableView滚动流畅的方案和思路?
    子线程加载数据,主线程刷新,cell复用,cell视图层级简化,尽量少触发离屛渲染;
  • 方案一

CPU把下面操作放到子线程:

  1. 对象的创建, 调整, 销毁
  2. 预排班(布局计算, 文本计算)
  3. 预渲染(文本等异步绘制, 图片编解码等)
  • 方案二
GPU优化
  1. 文理渲染
  • 1.1 避免离屏渲染
  • 1.2 依托于CPU异步绘制机制减轻GPU压力
  1. 视图合成
  • 2.1 如果多个View层层叠加, GPU就要做每个视图的合成, 合成每一个像素点的像素值, 需要大量的计算
  • 2.2如果减轻视图的复杂性, 就可以减轻GPU的压力, 也可以采用CPU异步绘制, 使得提交的位图本身就是一个层级很少的视图, 也可以减轻GPU的压力
  1. 什么是离屏渲染?
    指的是GPU在当前屏幕缓冲区以外新开辟一个缓冲区进行渲染操作

  2. UIView和CALayer之间的关系是怎样的?

  • UIView是专门负责事件传递和视图响应的, CALayer是专一负责视图显示的;单一职责设计原则(六大设计原则之一)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容