本文首发于我的个人博客
定义
- 可选项,一般也叫可选类型,它允许将值设置为nil
- 在类型名称后面加个问号? 来定义一个可选项
1 | var name: String? = "Jack" |
强制解包
- 可选项是对其他类型的一层包装,可以将它理解为一个盒子
- 如果为nil,那么它是个空盒子
- 如果不为nil,那么盒子里装的是:被包装类型的数据
1 | var age: Int? // 默认就是nil |
- 如果要从可选项中取出被包装的数据(将盒子里装的东西取出来),需要使用感叹号! 进行强制解包
1 | var age: Int? = 10 |
- 如果对值为nil的可选项(空盒子)进行强制解包,将会产生运行时错误
1 | var age: Int? |
判断可选项是否包含值
1 |
|
可选项绑定
- 可以使用可选项绑定来判断可选项是否包含值
- 如果包含就自动解包,把值赋给一个临时的常量(let)或者变量(var),并返回true,否则返回false
1 | if let number = Int("123") { |
eg:
1 | enum Season : Int { |
等价写法
可选项绑定中,如果多个条件比如下面
1 |
|
可以用 , 分割开,看起来更简单
1 | if let first = Int("4"), |
while循环中使用可选项绑定
- 有如下需求
//遍历数组,将遇到的正数都加起来,如果遇到负数或者非数字,停止遍历
1 | //遍历数组,将遇到的正数都加起来,如果遇到负数或者非数字,停止遍历 |
空合并运算符 ??(Nil-Coalescing Operator)
eg:
1 |
|
- a ?? b
- a 是可选项
- b 是可选项 或者 不是可选项
- b 跟 a 的存储类型必须相同
- 如果 a 不为nil,就返回 a
- 如果 a 为nil,就返回 b
- 如果 b 不是可选项,返回 a 时会自动解包
规律: 返回的类型取决于b
举例如下:
1 | let a: Int? = 1 |
1 |
|
1 |
|
1 |
|
1 |
|
多个 ?? 一起使用
1 |
|
1 | let a: Int? = nil |
1 | let a: Int? = nil |
??跟if let配合使用
1 |
|
1 |
|
guard语句
- 当guard语句的条件为false时,就会执行大括号里面的代码
- 当guard语句的条件为true时,就会跳过guard语句
- guard语句特别适合用来“提前退出”
1 |
|
- 当使用guard语句进行可选项绑定时,绑定的常量(let)、变量(var)也能在外层作用域中使用
假设我们有个登陆的需求,要求输入账号,密码。缺一不可。
用if语句书写
1 | //if语句实现登陆 |
如果用guard来书写
1 |
|
隐式解包(Implicitly Unwrapped Optional)
- 在某些情况下,可选项一旦被设定值之后,就会一直拥有值
- 在这种情况下,可以去掉检查,也不必每次访问的时候都进行解包,因为它能确定每次访问的时候都有值
- 可以在类型后面加个感叹号 ! 定义一个隐式解包的可选项
1 |
|
注意不能设置为nil
1 | let num1: Int! = nil |
尽量不要使用这个强制解包。
除非你设计接口,不希望接收空值,如果别人传Nil过来,直接报错
字符串插值
- 可选项在字符串插值或者直接打印时,编译器会发出警告
1 |
|
- 至少有3种方法消除警告
1 | print("My age is \(age!)") |
多重可选项
- 可以使用lldb指令 frame variable –R 或者 fr v –R 查看区别
1 | var num1: Int? = 10 |
还有下面这种
1 | var num1: Int? = nil |
参考资料: