手工改进XCode对Swift的响应速度

前言

​ Swift是一门非常优秀的语言。由于没有历史包袱,Swift得以集众家之长,甚至可以说激进。默认非空,类型安全,严格的编译时检查,为开发者避免了许多的坑。而强大的类型推断,面向协议的设计,以及函数式支持,使得写代码成为一种非常愉悦的体验。

​ 当然,作为一门只有4年历史的语言,Swift仍然有许多亟需改进的地方。对于大一些的项目,日常开发中经常会遇到代码高亮失效,代码提示失效的问题,好好的一个IDE活生生变成了文本编辑器。而论项目编译时间,Swift项目的编译速度通常是OC的3倍。这让用iMac(8G内存,笔记本机械硬盘)的笔者养成了经常盯着屏幕发呆的习惯。

​ 实际上,在Swift普及率较高的欧美开发者当中,也有人表示,对于大一些的项目,宁肯用OC。不是Swift语言不好,实在是XCode太坑(当然主要是swiftc这个前端的锅)。好在,Swift团队的一些成员正在重点改进编译相关问题,目前的测试数据似乎不错。预计到XCode10实装的时候,能够明显改进项目的编译速度,IDE变白板的情况也会改善很多。

正文

​ 在那之前,官方提供了一个工具,可以方便定位会产生编译瓶颈的代码。具体做法如下(XCode 8 以上):

  1. 在项目的target中,打开build setttings
  2. 在里面找到Other Swift Flags这一项
  3. 添加两个参数:-Xfrontend-warn-long-function-bodies=100,这里100单位是毫秒
  4. 重新编译项目

以上的设置,会在编译耗时过长的函数方法上产生一个警告:

Instance method 'updateCurrentTime()' took 6228ms to type-check (limit: 100ms)

​ 没错,6秒钟,而且只是类型检查😂。对于类型检查来说,100ms已经是很大的数字了,大部分情况下,1ms左右才是合理的。

简单说一下Swift代码编译过程:

  1. Parsing: 执行语法检查,生成AST语法树,返回IDE语法错误及警告

  2. Semantic analysis: 语义分析,执行类型检查及类型推断,生成类型完整的AST,返回IDE语义错误及警告。

  3. Clang importer、SIL generation等等。。。

​ 编译过程的前两步,直接决定了语法高亮和代码提示的响应速度。这个过程里Swift与OC最大的不同,便是引入了类型推断。实际上个人以为类型推断是Swift能够拥有简洁优雅语法的核心要素之一。想当年C++没有auto的时候,用Vector之类的容器真是让人欲仙欲死。

真是成也类型推断,败也类型推断。

我们来看看有问题的代码:

// _lastTime和currentTime都是Double类型的属性
guard abs(Int(round(_lastTime)) - Int(round(currentTime))) > 0 else {
            return
        }

其中abs()round()是对应C函数的泛型重载,Int()是参数重载,->是运算符重载。

看来复杂表达式中有了过多泛型和重载之后,类型推断的性能会指数级的下降,随之而来的便是类型检查时间过长。

因此,作为应对手段,我们只能把代码拆开:

let iLast: Int = Int(round(_lastTime))
let iCurr: Int = Int(round(currentTime))
let diff: Int = abs(iLast - iCurr)
guard diff > 0 else { return }

于是警告消失了,通过拆分表达式和显示类型提示,这段代码的类型检查速度提高了至少60倍。也由于这是代码编译过程的一部分,每次重编译代码都节省了5秒钟的人生。

这是一个悲伤的故事😕。

上面的例子可能比较特殊,不过除了重载以外,尾随闭包也是类型推断的重灾区。比如下面的代码:

// 250ms type-check
let averageScore = Int(round(scoreResult
     .map{Float($0.value)}
     .reduce(0, +)
     / Float(scoreResult.count)))

还是拆代码,以及闭包的显式类型提示:

// 10ms 以内
let ast: Float = scoreResult
    .map{r in return Float(r.value)}
    .reduce(0, +)
    / Float(scoreResult.count)
let averageScore: Int = Int(round(ast))

这么把警告一个个干掉之后,虽然不可能完全解决问题,但是满屏黑白的情况会减少很多,至少不用经常习惯性的按⌘S来刷新了。(没错,除了代码键入,保存也能触发语法检查)

后记

在官方对于swift编译性能改进的分析中(链接),提到了一些改进方向:

  • 增量编译模型过于保守,许多不必要的重新编译。
  • 名字解析过于激进,读入(反序列化)了过多的定义。
  • 前端任务的二次方级(复杂度)的任务中,很多引用的定义的类型检查不够lazy。
  • 表达式的类型推断对于约束的推导没有效率,某些情况下(时间复杂度)是超线性甚至指数级的。
  • SIL优化的分析过程有时会缓存失败,导致超线性的性能退化。
  • SIL生成IR的过程,在某些情况下(比如大的值类型)会生成过多的IR码,影响后端LLVM的处理时间。

​ 由此,我们可以期待,在Swift 5发布的时候,无论是IDE的性能,还是代码编译速度,都有一个明显的改善。

PS:Swift 5还可以期待async/awaitOwnership这样的并发及运行时性能改进,以及官方的主要目标,ABI稳定(跨版本二进制兼容的一部分)。

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