tableView性能优化

tableview的优化一直是一个很考验基本功的活儿,之前做项目的适合被这个问题困扰了很久,通过性能工具、查阅文档解决,整理思路和解决方案如下:

tableview优化最主要:复用cell,header,footer实例;使用约束布局cell子控件时不多次添加约束;图片不过大,尽量不使用透明视图;避免阻塞主线程;计算高度方法不做大量逻辑处理。
  • cell是否使用了复用机制而不是每一次都创建新的cell。

如果每次都创建新的cell,在滑动的时候会表现为:刚开始的时候很顺畅,但是会越来越卡,内存跟着一直升高,停止滑动的时候也不会降下来。使用缓存机制创建的cell,开始滑动的时候内存会开始上升,等创建了一个屏幕再加半屏的cell之后,内存趋于平稳。

  • cell是否添加了大量的子控件,或者对layer做了过多的操作。

如果添加了大量的子控件,使用drawRect方法添加子控件,平衡GPU与CPU的负担。同时还需要注意尽量使用不透明视图和不重叠的渐变,否则会加大GPU的负担,造成性能不佳。

补充:请谨慎使用drawRect:方法,当cell中只有少量的子视图时,应当避免使用,因为重写drawRect:就是在此方法中一句代码都不写也会占用5-6M的内存。
  • 高度计算方法时不做复杂的计算,尽量只使用加减乘除。

自适应高度的cell实现方式有很多种,比如,
1.使用iOS7以上系统的

func tableView(tableView: UITableView, estimatedHeightForRowAtIndexPath indexPath: NSIndexPath) -> CGFloat

这个方法中,可以先给一个估计的高度,系统会从你给定的高度再去计算实际高度。但是在使用过程中会出现cell突然变高变得低的情况,不适用于高度变化太大的cell。
2.如果使用约束布局创建的cell子控件,子控件之间都建立了相互约束,最上面的子控件与cell顶部建立约束,最下面的子控件与cell底部建立了约束,相当于子控件把cell撑开了。


约束简图

这时在高度计算方法中,走一遍cell的loaddata方法后可以通过

func systemLayoutSizeFittingSize(targetSize: CGSize) -> CGSize 

取得cell的size,进而得到cell高度。
通过这个方法获取的cell高度是十分精确的,只要创建好子控件的约束就能获得cell的size。比较不好的是只是这种方法会重走一遍cell的loaddata方法。除此之外在调用cell的loaddata之前需要得到cell的实例,实例创建的方式应该与cellForRow方法一样,优先从缓存池中取得。
这个方案可能会创建多个cell。如果能在内存汇总保存一份cell的实例就能解决这个问题了!我讲讲我实现的思路:
首先先注册cell,当缓存池中没有cell时系统会自动创建,有的话会直接取缓存中的cell返回给你。

override func viewDidLodad() {
      tableView.registerClass(CardCell.self, forCellReuseIdentifier: ID)
}

用lazy创建一个cell实例,由于lazy 关键字,cell的创建只会执行一次

lazy var cell:CardCell = {
        //已经注册过cell,当缓存池中没有cell时系统会自动创建,有的话会直接取缓存中的cell返回
        let v = self.myTableView?.dequeueReusableCellWithIdentifier(self.ID) as! CardCell
        return v
        }()

通过懒加载的方式,只创建一次cell的实例,避免内存浪费。
接下来要做的步骤就是之前讲的,调用cell的loadData方法,计算高度

func tableView(tableView: UITableView, heightForRowAtIndexPath indexPath: NSIndexPath) -> CGFloat {
    self.imageCell.loadData(d)    
    let height:CGFloat = self.cell.contentView.systemLayoutSizeFittingSize(UILayoutFittingCompressedSize).height
return height
}

之前查资料的时候还有用空间换取时间的方案:
1)在请求网络数据成功后就计算好高度并通过字典或者数组保存高度值,在高度方法中直接根据数组下标或者key值取得高度并返回。
2)还有建立一个frameModel的方法,与1中相似,只是获得网络数据后保存到frameModel中,在frameModel中定义一个类方法,通过获得的model值计算高度后返回。

  • 避免快速滑动情况下开过多线程。

cell中的图片开线程异步加载,相信每个人都会想到。线程开过多了会造成资源浪费,内存开销过大。图片过多时可以不要一滚动就走cellForRow方法,可以在scrollview的代理方法中做限制,当滚动开始减速的时候才加载显示在当前屏幕上的cell(通过tableview的dragging和declearating两个状态也能判断)

  func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        var canLoad:Bool = !tableView.dragging && !tableView.declearating
        if  canLoad {
            //开始loaddata,异步加载图片
      }
}
  • 图片处理

1)后台下载图片后再回主线程刷新UI,避免阻塞主线程。
2)图片过大回造成GPU负担过大,可以在图片下载后压缩尺寸后显示
3)避免对layer做过多的操作,尽量设置图片为不透明

补充:
  • 简单的设置cornerRadius是不会影响性能的,但是设置了maskToBounds,会导致离屏渲染,应减少设置图层 maskToBounds = YES ,;
  • 使用懒加载图片的方式避免重复下载图片,浪费资源。图片下载后并做压缩处理后将其保存到缓存中,下次加载此图片之前先从缓存中取,如果取不到该图片就在后台下载保存。
  • 使用Core Graphics实现圆角等功能。
  • 重写drawRect方法会离屏渲染,导致内存急剧上升,即使在这个方法里面不写一句代码,也会让内存升高。

参考:
iOS 高效添加圆角效果实战讲解
消息别让圆角成为你的列表杀帧高手
内存恶鬼DrawRect
关于性能的一些问题

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

推荐阅读更多精彩内容

  • tableview的优化一直是一个很考验基本功的活儿,之前做项目的适合被这个问题困扰了很久,通过性能工具、查阅文档...
    青葱烈马阅读 4,444评论 1 7
  • 一、tableview通用工作流程 一个tableView需要显示内容的时候,首先会发送网络请求,向服务器请求数据...
    强降雨天气阅读 858评论 0 5
  • TableView相信只要是做iOS开发的就不会陌生,目前大多数iOS的app都是采用TabBar+Navigat...
    90后的思维阅读 728评论 0 6
  • 生活不会永远按照我们的意愿进行。 孩子也不可能按照我们的意愿来活。遇到问题,在所难免。 正面管教这一章强调关注于解...
    叶子暖洋洋阅读 313评论 0 0
  • 我不懂夸父为何一直逐日 可能我习惯了日出日落 我不懂精卫为何一直填海 可能我习惯了四季变换 我不懂 直到遇见了你 ...
    哆啦A念阅读 172评论 0 0