博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
如何让tableView保持顺畅
阅读量:6655 次
发布时间:2019-06-25

本文共 1388 字,大约阅读时间需要 4 分钟。

UITalbeView作为一个项目中常用的开发组件之一,如何让其保持流畅的滑动则是最重要的一个环节。

UITableView的核心思想为cell的重用机制。简单的理解就是将滑出屏幕外的cell放进重用池,当要显示某一位置的cell时,先去检查重用池中是否有可重用的cell,如果有,就直接拿来显示;如果没有,才会创建。这样做的好处可想而知,极大的减少了内存的开销。

然而重用cell只是优化UITableView的基础。

病因

App 主线程开始在 CPU 中计算显示内容,比如视图的创建、布局计算、图片解码、文本绘制等。随后 CPU 会将计算好的内容提交到 GPU 去,由 GPU 进行变换、渲染等一系列的处理。随后 GPU 会把渲染结果提交到帧缓冲区去,等待显示到屏幕上。由于垂直同步的机制,如果一定的时间内,CPU 或者 GPU 没有完成内容提交,则那一帧就会被丢弃,等待下一次机会再显示,而这时显示屏会保留之前的内容不变。这就是界面卡顿的原因。

占用资源主要原因及解决方案

1、控件的创建

在tableView中肯定有大量的内容需要进行展示,所以在cell中肯定需要创建大量的UIView、UILabel之类的控件,在创建时就需要分配内存、调整属性,这些操作都是比较消耗CPU资源的。

在cellForRowAtIndexPath中不要出现创建对象的代码,尽量在cell初始化的时候创建好所有的控件。尽量使用更为轻量的CALayer来代替UIView。

2、视图布局的计算

视图布局的计算是吃CPU资源的主要原因之一,因为计算,所以这方面的资源占用率就不可避免了。

虽然是不可避免的,但也需要将占用资源比例降低。

不论通过何种技术对视图进行布局,其最终都会落到对 frame等属性的调整上。对这些属性的调整非常消耗资源,所以尽量提前计算好布局,在需要时一次性调整好对应属性,而不要频繁的计算和调整这些属性。

3、文本的计算

上面说了只要有计算,对资源的占用就不可避免,所以将文本宽高的计算放入子线程操作,会给主线程减少相当一部分的空间。

4、图像的绘制与图形的生成

drawRect对资源的占用大家应该都有所了解,因CoreGraphic 方法通常都是线程安全的,所以图像的绘制可以很容易的放到后台线程进行。

在项目中经常会把头像之类的控件设置成圆角,如果我们操作CALayer的属性通常都会触发离屏渲染,而当一个界面中出现大量的圆角图像时,滑动帧率就会降得很低。所以尽量避免使用CALayer来生成圆角属性。设置圆角可以参考:

优化技巧

预排版

从后台的回调中拿到数据时,通过后台线程将所有控件的高度与cell的整体高度都给计算出来,然后将这些计算出来的高度都给缓存起来。

高度返回

若高度一定,直接使用rowHeight属性而不是使用heightForRowAtIndexPath方法,以减少调用的消耗。

图片设置

在设置显示图片时,不要直接设置UIImageView的contentMode属性自动适应,图片变形会计算transform,压缩时会乘以一个矩阵,消耗性能。对于要求性能较高的app,应该将得到的图片经过处理成UIImageView大小后再呈现。

转载于:https://juejin.im/post/5a6821866fb9a01c91408e83

你可能感兴趣的文章
html input type=file 选择图片,图片预览 纯html js实现图片预览
查看>>
相机上的P,S,A,M分别是什么单词的缩写?
查看>>
杭州电子科技大学2018年自命题科目考试大纲(数据结构与组成原理)
查看>>
django 的缩略图sorl-thumbnail的使用连接地址
查看>>
WPF:如何为程序添加splashScreen?
查看>>
用js生成PDF的方案
查看>>
[LeetCode] K-th Smallest Prime Fraction 第K小的质分数
查看>>
鹅厂优文 | ReactJS一点通
查看>>
安装Gradle(Windows & Linux)
查看>>
centos7的时间同步机制:chrony使用
查看>>
answerOpenCV轮廓类问题解析
查看>>
Windows 快捷键总结
查看>>
网站运维工具使用iis日志分析工具分析iis日志(iis日志的配置)
查看>>
fusionjs uber开源的通用web插件化开发框架
查看>>
mongodb批量操作, bulk_write,
查看>>
SVG 图像入门教程
查看>>
java如何实现python的urllib.quote(str,safe='/')
查看>>
C++ 使用 hiredis 封装redis 的数据获取接口
查看>>
python mysql redis mongodb selneium requests二次封装为什么大都是使用类的原因,一点见解...
查看>>
分析轮子(四)- 我也玩一把 Serializable.java
查看>>