重读 Swift 之一:Optional(可选型)

之前学习 Swift 的时候都是比较破碎,零零散散,以至于有些地方学习的不够透彻。所以趁最近有时间,就开始重新学习 Swift ,从最基本的开始学习,希望能深入理解,见微知著!
关于 Swift 我们都是到是类型安全型的,相较于 OC ,安全似乎更能体现出来。关于这一方面, Swift 的可选型(optional)就不得不说了,可选型可以说是 Swift 最为突出的特性之一。可能有的同学在看 Swift 的过程中会遇到可选型(比如 String? ),有时也会见到 String! 这样的,搞不明白其中的意思,所以今天就来一步步理解可选型。

一、什么是 Optional


Optional 是 Swift 新加入的类型,所以学过 OC 的童鞋就知道在 OC 中是没有这个概念的。可选型的意思简单理解是:有值得时候就有值,无值的时候就是 nil 。Swift 中的 nil 和其他语言中的 nil 还有些不一样,nil 自己本身就是一种类型,没有就是 nil ,是和其他类型严格区分开的。
可选型的定义很简单:类型 + ? 。比如 String?Int?Float? 等,这里需要注意的是 String?String 是完全不同的两个类型,前者是 String 类型的额可选型,后者是 String 类型,注意区分。举个例子,在 OC 中我们可以这样写

 NSString *name = @"loveway";
 name = nil;

上面的这种写法在 OC 中是没有问题的,相比于上面,在 swift 中

var name: String = "loveway"
name = nil

如果我们像上面代码中这样写就会报错,如下

error

说的是 nil 是不可以分配给 String 类型的,这也说明在 swift 中 nil 是和其他类型严格区分的。改为可选型即可,

var name: String? = "loveway"
name = nil

可选型,顾名思义就是可以选择,比如 String? 的意思就是可以在 Stringnil 之间选择,可以是 String 也可以是 nil 。如果一个变量定义成 String,那么这个变量就会是 String 类型,而不可能是nil
还有一点需要注意的是声明可选型必须是显示的声明也就是必须是

var name: String? = nil

这样,而不能是

var name = nil

这样,因为 Swift 在做类型判断的时候无法判断 name 到底是 String 类型的可选型还是其他类型的可选型,因此会导致编译错误。

Tips: 当然枚举类型的写法还可以如

var name: Optional<String> = Optional.Some("Loveway")

这样,只不过我们一般都用

var name: String? = nil

这样的简写。

二、Optional的解包


可选型是不能够被直接使用的(因为 Swift 是类型安全的,可选型的值又可能会是 nil,如果不做处理可能导致程序 crash),如果我们想使用可选型的值,那么在这之前我们需要做的一项工作就是:解包(unwarp)!

  • 1、强制解包
    所谓的强制解包意思就是我知道这个类型是可选型,但是在我的程序执行到这里的时候我可以保证它是有值得,所以我要在这里使用它。具体表现形式就是在可选型后面加个 !,如下
var name: String? = "loveway"
"My name is " + name!

但是这样的解包是不安全,因为你不知道什么时候你的这个可选型就会变成 nil,如果我们代码非常多的话,一不小心为 nil了,可能会导致程序崩溃。这个时候我们会想到一种方法:判空!如下,

var name: String? = "loveway"
if name != nil { 
   "My name is " + name!
} else {
    print("name is nil")
}

这样写似乎是没有什么问题了,但是需要注意的是,你在判断非 nil 的作用域内使用 name 的时候还必须把 ! 带上,这样代码比较多的时候还是比较麻烦。于是我们可以使用下面这种方式,

  • 2、使用 if let 解包
    使用 if let 解包如下,
var name: String? = "loveway"
if let name = name {
   "My name is " + name
} else {
    print("name is nil")
}

这种解包方式可以保证 name 是解包过的,不会再是 nil 这种情况,其实逻辑是和上面做非空判断一样的。当然你把 let 换成 var 也是可以的,效果是一样的,只不过我们一般要用的是解包后的值,而不会去改变它,所以平常使用中一般都是用 if let
同时 if let 可以同时一次性解包多个可选型,用 , 隔开,使语句简洁,如下

var name: String? = "loveway"
var age: Int? = 18
if let name = name, age = age {
   "My name is " + name
} else {
    print("name is nil")
}

最后,既然这里使用的是 if ,那么同样我们可以如下这样用,来进行进一步的判断筛选

var name: String? = "loveway"
var age: Int? = 18
if let name = name, age = age where age == 18 {
    print("My name is \(name), age is \(String(age))")
} else {
    print("name is nil")
}

如上也是可以的。

三、可选链式调用(Optional Chaining)


可选链式调用(Optional Chaining)是一种可以在当前值可能为 nil 的可选值上请求和调用属性、方法及下标的方法。如果可选值有值,那么调用就会成功,如果可选值是 nil ,那么调用将返回 nil。多个调用可以连接在一起形成一个调用链,如果其中任何一个节点为 nil ,整个调用链都会失败,即返回 nil 。
下面我们就来举个例子具体说明可选链,如下

var name: String? = "loveway"
if let name = name {
    print(name.uppercaseString)
} else {
    print("name is nil")
}

我们来解包 name ,如果有值就打印出 name 的大写,如果没有就输出 name is nil ,其实上面这段代码完全等同于

var name: String? = "loveway"
name?.uppercaseString

上面这句代码的意思就是如果可选型变量 name 有值,那么就对 name 进行解包,并得到 name 的uppercaseString 值,如果没有,那么这句代码就会返回一个 nil 。这样就符合 Swift 的类型安全,完全是没有问题的。但是如果你写成

var name: String? = "loveway"
name!.uppercaseString

也是可以的,不过不安全,因为如果 name 没有值,你进行强制解包,就会报错。
所以我们以后可能会用到类似于 person?.name?.uppercaseString 这样的一层层解包的,这种就是可选链。

四、Nil Coalescing Operator(空合运算符)


如上,如果我们想把解包后的值存起来的话,可以这样

let newName = name == nil ? "no name" : name!

上面代码的意思就是如果 name 为 nil ,newName 就是 "no name",否则 newName 就是 name!(name的解包)。其实 Swift 为我们提供了更简洁的语法,如下

let newName2 = name ?? "no name"

上面代码的意思就是 name 如果有值 newName2 的值就是 name! ,否则就是 no name 。这里需要注意的是 ??是空合运算符,这样写的可阅读性强,比较简洁。
当然关于 ?? 远不止这些,有兴趣的同学可以看 聊聊swift语言中的“??” 这篇文章。

五、隐式可选型


上面我们都知道了,创建一个显示可选型是: 类型 + ?。这里创建隐式可选型的就是: 类型 + !

var name: String! = "loveway"

这里可能有的童鞋会疑惑,已经有了显示的可选型,为什么还需要有隐式的可选型。这里其实隐式的可选型也是有一定作用的。比如你有一个变量,声明为隐式的可选型(!),它的作用就是,当你这个类没有被初始化的时候他是没有值的,但是当你这个类初始化以后,你可以确保他是有值的,所以这里声明为隐式的可选型,而不是显示的。同样需要注意的是隐式的可选型也是可选型,如果你需要用它的值,你也是要进行判断的。如果不进行判断而直接使用,可能会造成不可预料的后果!
差不多可选型就到这里了,如果还有什么遗漏,欢迎大家指正!

参考链接:可选链式调用(Optional Chaining)

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

推荐阅读更多精彩内容