《高性能iOS应用开发》用户界面

应用以 60 帧每秒运行,意味着有 16 毫秒事件进行下一帧过渡的全部操作,因此要在一个时间循环中将子任务在主线程的累积时间消耗缩到最短。

ViewController

ViewController 生命周期如下图所示

ViewControllerLifeCycle.png

最佳实践:

  • 保持 ViewController 轻量,ViewController 只是纽带,不能存放业务逻辑
  • 不要在 ViewController 中编写动画逻辑,动画可以在单独的动画类中实现,该类接受视图作为参数,返回动画给 ViewController,然后添加至视图上
  • 使用数据源和委托协议,将代码按照数据检索、数据更新和其他业务逻辑进行分离,ViewController 只负责选择视图并连接到供应源上
  • ViewController 负责响应来自操作系统的 UI 相关事件,包括旋转和内存警告
  • 不要编写自定义 init 方法,如果 ViewController 切换至 XIB 或故事板则 init 方法就不会被调用了
  • 不使用代码手工布局 UI
  • 创建一个实现了公共设置的基类 ViewController
  • 在各个子 ViewController 中,使用 category 创建可复用的代码

当 ViewController 的 view 被请求时,loadView 方法会被调用,此时 view 为 nil。

[sampleViewcontroller view];

执行过程中应该尽量缩短在 viewDidLoad 上花费的时间,数据应该提前准备好或者在其他线程进行加载。

视图结构和渲染包括以下步骤

  • 构造子视图
  • 计算并提供约束
  • 为子视图递归执行步骤 1 和步骤 2
  • 递归渲染

视图可见性

视图可见性有四个生命周期方法

  • viewWillAppear

此时页面过渡动画还没开始,视图对用户不可见,启动任何动画都不会生效。

  • viewDidAppear

过渡动画大约 300 毫秒。此时可以启动或恢复动画。

  • viewWillDisappear

视图被覆盖或视图被弹出时触发。

判决方法

    NSInteger index = [self.navigationController.viewControllers indexOfObject:self];
    if (index == NSNotFound) {
        NSLog(@"willDisappear 即将出栈");
    } else {
        NSLog(@"willDisappear 被覆盖");
    }
  • viewDidDisappear

此时视图已经被从 navigationController 中移除。

视图可见性的最佳实践:

  • 不要重写 loadView
  • 将 loadView 作为最后检查点,查看数据源是否可用并更新 UI
  • 选择性考虑在 viewWillAppear 中更新 UI
  • 在 viewDidAppear 中开始动画或视频播放
  • 在 ViewWillDisappear 中停止动画
  • 在 viewDidDisappear 中销毁数据结构

View

视图优化基本规则:

  • 尽量减少主线程工作
  • 避免较大的 nib 或故事板,xml 文件在真正使用前需要加载和解析,应该尽量模块化和最小化,从而加快速度,减小内存占用
  • 避免多层嵌套,在层次结构任何位置添加 View 时,祖先树节点会执行 layoutSubviews,这个调用代价大,因为 view 需要根据约束重新计算子视图位置,并在视图层级的每一次都进行此操作
  • 尽可能延迟加载 view 并重用,如果需要可以创建自己的视图缓存。
  • 复杂 UI 最好使用自定义绘图,因为这样只会触发一个视图进行绘制,同时避免了调用代价较高的 layoutSubviews 和 drawRect 方法

UILabel

UIlabel 的渲染过程如下

  • 计算需要的像素数目,消耗较大
  • 检查要渲染的宽度
  • 检查 numberOfLines,计算将要展示的行数
  • sizeToFit 是否被调用,是则计算高度
  • 如果没有调用 sizeToFit,则检查当前内容能否在给定高度下展示出来
  • 如果不能,则根据 lineBreakMode 确定截断位置
  • 使用字体、类型及颜色渲染最终展示的文本

如果动态计算出的 UILabel 宽度是容器宽度的一部分,那么要保证宽度可以由一个百分比均匀分配,否则渲染时需要进行反锯齿操作,代价很大。

UIImageView

最佳实践:

  • 对于多次使用的图片,imageNamed 方法可以缓存到内存中
  • 对于只使用一次的大图片,考虑使用 imageWithContentsOfFile 方法,避免缓存到内存中
  • 使用高性能图像缓存库,根据 RAM 百分比来确定缓存参数
  • 尽量使载入的图片与目标 UIImageView 大小相互匹配,否则调整图片大小会耗费性能
  • 对于需要模糊或色调的效果的图片,可以创建一个图片的副本,将效果添加到副本上
  • 加载图片要在统一专用的一个非主线程队列中执行
  • 尽量使用绘制自定义 view 的方式替代图片的覆盖方式

UITableView

最佳实践:

  • 重用 cell,避免回收和重复创建 cell
  • 避免动态 cell 高度,如果需要可变高度的 cell,则尽量选择较高的 cell,从而为较少的 cell 计算高度
  • 如果真的需要动态高度的 cell,用一个规则来标记需要计算并缓存高度的单元格
  • 用自定义 view 重用 cell 时,避免每次都调用 layoutIfNeeded 来布局,固定元素的尺寸,可以确保 cell 渲染所需时间最小
  • 避免透明的 cell 子 view
  • 使用占位壳视图在快速滚动时进行 cell 占位,当滚动速度降低到阈值以下时再刷新数据
  • 避免渐变、图像缩放、以及任何屏幕外的绘制

UIWebView

  • 尽可能复用 webview,同时避免内存泄漏;展示新的 URL 之前先将内容重置,在 loadRequest 方法后调用 loadHTMLString:baseURL
  • 附加一个自定义的 UIWebViewDelegate,实现 webview:shouldStartLoadWithRequest:navigationType 方法,从而获知用户正在逃离应用
  • 通过 stringByEvaluatingJavaScriptFromString 建立桥来连接应用和 JS
  • 实现委托的 webView:didFailLoadWithError: 方法
  • 实现 webView:didFailLoadWithError: 方法来处理特定错误
  • 对 HTTP 协议错误进行处理

自定义视图

自定义视图可以采取复合视图或直接绘制两种方式,直接绘制的性能高,但是维护困难,适合稳定下来的模块。

自注:鉴于利用 drawRect 方式调用 CoreGraphic 方法是在 CPU 中开辟上下文进行绘制后,整个内存区提交给 GPU 渲染,其内存占用会暴增,而从速度角度来讲,创建和切换上下文的时间消耗,与 UIKit 对渲染绘制的优化可能不分伯仲,不能有效论证直接绘制方案一定是最优的,因此能采用 UIKit 尽量采用 UIKit 方式。

自动布局

自动布局使用的 Cassowary 算法复杂度为 O(N),其中 N 为约束数目。自动布局性能上比手动设置 frame 要差。

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

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,067评论 4 62
  • 踏~踏~踏~一个缓慢略带轻浮的脚步声从远处的巷口传来,一盏老久满是锈迹的路灯,残烛般昏黄的灯光忽明忽暗,没有给这陈...
    黑羽雨秋阅读 234评论 0 1
  • 下雪了,好大好大的雪啊! 天儿冷了,希望疼我爱我的亲人注意保暖! 最最大的幸福莫过于希望家人幸福安康!情同手足独在...
    军缘情阅读 181评论 0 0
  • 偷得浮生半日闲,早入书馆阅恺书。 娓娓道来详故事,晓人晓域晓一生。 这本书朴素而别致,淡淡的花纹,淡淡的文字,淡淡...
    问蹊阅读 264评论 0 0