Swift3新特性(What's New in Swift)

前言

Swift3即将发布release版本,WWDC2016上苹果也带来了介绍.点击这里看视频.本文主要是翻译和解释.另外原视频前面介绍了Swift是怎么开源的,最后谈到了如何提高效率.对于一般开发者并无大用,我这里也就不多说了.

API简化

许多API是从OBjective-C版本直接转成Swift版本,所以不够Swift化.苹果在Swift3中简化了大量API,使其语义更清晰,语法更简洁.举例如下:

// Swift 2
array.appendContentsOf([2,3,4])
array.insert(1, atIndex: 0)
// Swift 3
array.append(contentsOf: [2,3,4])
array.insert(1, at: 0)

// Swift 2
if url.fileURL {}
x = url.URLByAppendingPathComponent("file.txt")
// Swift 3
if url.isFileURL {}
x = url.URLByAppendingPathComponent("file.txt")

Objective-C API优化

1. Swift化
Objective-C中存在着大量的C式API,在Swift中显得格格不入.在Swift3中得以改进.如:

// Swift 2
func CGContextFillPath(_: CGContext)
// Swift 3
void CGContextFillPath(CGContextRef) NS_SWIFT_NAME(CGContext.fillPath(self:));
extension CGContext {
  func fillPath()
}

2. Objective-C泛型支持
Objective-C本身是不支持泛型的,当我们使用某些通用的API是就相当痛苦.比如使用Core Data查询得到的结果,明明可以指定类型.在Swift3中是这样改进的:

// Swift 2
func findAnimals() {
  let request = NSFetchRequest(entityName:”Animal")
  guard let searchResults =
       try? context.executeFetchRequest(request) as! [Animal] {
    return
  }
  ...
  use(searchResults)
}
// Swift 3
func findAnimals() {
  let request : NSFetchRequest<Animal> = Animal.fetchRequest
  guard let searchResults = try? context.fetch(request) {
    return
  }
  ...
  use(searchResults)
}

这里的guard语句的else漏掉了,我怀疑是演讲稿有问题,并不是修改了

3. String型的Enum改进
在Objective-C中String型的Enum到处可见(与C语言相关).而String的不安全性很大.使用者可以随意传入String从而导致Bug.在Swift3中对此也进行了优化.

// Swift 2
typedef NSString *NSNotificationName;
const NSNotificationName NSUserDefaultsDidChangeNotification;
let NSUserDefaultsDidChangeNotification: String  //  Imported definition
center.addObserver(forName: NSUserDefaultsDidChangeNotification, ...) // Use

// Swift 3
typedef NSString *NSNotificationName;NS_EXTENSIBLE_STRING_ENUM; 
const NSNotificationName NSUserDefaultsDidChangeNotification;
extension UserDefaults {
  class let didChangeNotification: NSNotification.Name
}  //  Imported definition
center.addObserver(forName: UserDefaults.didChangeNotification, ...) // Use

Core Language(语言特性)

1. 参数标签
首先就是方法的首个参数名称不再默认略去(这是Objective-C的命名特点).如:

func myFunction(a: Int, b: Int, c: Int) { } 
// Swift 2
myFunction(42, b: 57, c: 99)
// Swift 3
myFunction(a: 42, b: 57, c: 99)

2. 泛型约束
泛型的约束条件过长引起阅读困难,将where部分挪至函数尾部.如:

// Swift 2
func anyCommon<T: Sequence, U: Sequence where T.Element: Equatable, T.Element == U.Element>(lhs: T, rhs: U) -> Bool {
// Swift 3
func anyCommon<T: Sequence, U: Sequence>(lhs: T, rhs: U) -> Bool where T.Element: Equatable, T.Element == U.Element {

3. 未使用返回值警告
现在允许将函数标记为@discardableResult用于不使用返回值且不警告.

@discardableResult
func plusOne(_ a: Int) -> Int {
  print(a)     // side effect!
  return a+1
}
plusOne(4) // no warning

类型系统

1. UnsafeMutablePointer
在Swift2中可以将nil直接赋给UnsafeMutablePointer,苹果原本就是把UnsafeMutablePointer对待成不安全的指针.但是Swift3中苹果不愿再将其特殊化, UnsafeMutablePointer将与其他类型一样有可选型.

// Swift 3
let ptr : UnsafeMutablePointer<Int> = nil
let ptr : UnsafeMutablePointer<Int>? = nil

2. 隐式可选型
隐式可选型为用!修饰的类型,它可以为nil,而且可以传给要求非nil的参数.不过当访问此值时会强制解包,如果为nil则会引起崩溃.Swift3中针对非可选型与隐式可选型相加得到的结果类型进行了改变.

func f(value : Int!) {
  let x = value + 1 // x: Int - force unwrapped 
  let y = value // Swift2中 y: Int!, Swift3中 y: Int?
}

标准库

1. Collection的新Index
Swift对于Index的处理一直让人感觉捉襟见肘,Swift3中对于语法进行了改进.具体如下:

// Swift 2
i = collection.startIndex
next = i.successor()
// Swift 3
i = collection.startIndex
next = collection.index(after: i)

其他的改进还有:

  • 半开区间和区间并入Range型
  • 0...UInt8.max现可以正确生效
  • 性能提升

2. 浮点数和数值
浮点数包含Float,Float80,Double和CGFloat.我想很多人应该疲于浮点型间的类型转换以及运算.在Swift3中pi值可以在任何浮点型中get到.而且浮点型间的运算无需再频繁转换.代码如下:

let v = 2 * Float.pi                 // 轻松get pi值
return x * CGFloat.pi / 180   // Float间运算无需转换

附录

关于新版本的改动还有非常多,有一些是一笔带过的,还有更多是没有提到的.在这里我以附录的形式呈现给大家,也方便以后补充.

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

推荐阅读更多精彩内容