枚举的基本用法
等价于 =>>
使用:
-------------------------------------------------------------------------------------------------------------------------------
关联值(Associated Values)
有时会将枚举的成员值跟其他类型的关联存储在一起,会非常有用
- 必要时let也可以改为var
- 关联值举例
-------------------------------------------------------------------------------------------------------------------------------
原始值(Raw Values)
-------------------------------------------------------------------------------------------------------------------------------
隐式原始值(Implicitly Assigned Raw Values)
-------------------------------------------------------------------------------------------------------------------------------
递归枚举(Recursive Enumeration)
枚举里的枚举类型中又调用了枚举,就是递归枚举,注意要在前面加上indirect,否则报错
等价于==>
结果是 5 + 4 - 2 = 7
-------------------------------------------------------------------------------------------------------------------------------
MemoryLayout
可以使用MemoryLayout来获取数据类型占用的内存大小
这里要注意pwd为33是因为要多出来一个字节来区分other和上面的number。
.stride : 分配占用的空间大小,必须是内存对齐参数alignment的整数倍,所以是40
.alignment : 内存对齐参数
.size : 实际用到的空间大小,这里的33表示number(Int, Int, Int, Int)中的Int每个占8个字节,总共占32个,other占一个字节,加起来33个字节。
这个占用四个字节
这里要说一下关联值和原始值:
number(Int, Int, Int, Int) : 属于关联值,将传进来的关联值存储到枚举变量里面,因为关联值由外部传入,可以任意变化,需要存储,而原始值固定默认的,不可改变
而.other : 属于原始值,不会存储到枚举变量内存里面的
-------------------------------------------------------------------------------------------------------------------------------
可选项(Optional)
注意:Swift里一般的值是不可以直接赋值为nil的,但是可选类型的值可以。
- 可选项,一般也叫可选类型,它允许将值设置为nil
- 在类型名称后面加个问号?来定义一个可选项
-> int ? 表示返回值可以为Int也可以为nil
-------------------------------------------------------------------------------------------------------------------------------
强制解包
- 可选项是对其他类型的一层包装,可以将它理解为一个盒子
- 如果为nil,那么它是个空盒子
- 如果不为nil,那么盒子里装的是:被包装类型的数据
- 如果要从可选项中取出被包装的数据(将盒子里的东西取出来),需要使用!进行强制解包
var age : Int? = 10
var ageInt : Int = age!
ageInt += 10
- 如果对值为nil的可选项(空盒子)进行强制解包,将会产生运行时错误
- 判断可选项是否包含值
-------------------------------------------------------------------------------------------------------------------------------
可选项绑定(Optional Binding)
- 可以使用可选项绑定来判断可选项是否包含值
- 如果包含就自动解包,把值给一个临时的常量(let)或者变量(var), 并返回true,否则返回false
备注:Season(rawValue: 6)是指取原始值为6的枚举,2就会取到summer
- 等价写法
注意:可选项绑定的并列条件不能用 && ,可以用,
-------------------------------------------------------------------------------------------------------------------------------
while循环中使用可选项绑定
-------------------------------------------------------------------------------------------------------------------------------
空合并运算符??(Nil -Coalescing Operator)
定义:
- a ?? b
- a 是可选项
- b 是可选项 或者 不是可选项
- b 跟 a 的存储类型必须相同
- 如果 a 不为nil,就返回 a
- 如果 a 为nil,就返回 b
- 如果 b 不是可选项,返回 a 时会自动解包
- 举例:
- 多个 ?? 一起使用
不可以写成 let c = a ?? 3 ?? b,因为 ?? 左边必须是可选型,不能是3
-------------------------------------------------------------------------------------------------------------------------------
空合并运算符??跟 if let配合使用
- if语句实现登陆
标注:if let tmp = info["username"]中的tmp作用域只在它大括号里,不能作用在括号外,所以要定义一个外部常量,来获取这个值
注意:字典类型的取值比如info["username"]这种字典类型的取值为可选类型(?),因为如果key值写错,取到的值就为nil,这时就要用到可选类型,但是数组当取-1这种超出取值范围的错误值的时候,编译器会直接报错,不会自动转为可选值
-------------------------------------------------------------------------------------------------------------------------------
guard语句
- 当guard语句的条件为false时,就会执行大括号里的代码
- 当guard语句的条件为true, 就会跳过guard语句
- guard语句特别适合用来"提前退出"
- 使用guard语句进行可选项绑定时,绑定的常量(let)和变量(var)也能在外层作用域中使用
-------------------------------------------------------------------------------------------------------------------------------
隐式解包(Implicitly unWrapped Optional)
- 在某些情况下,可选项一旦被设定值之后,就会一直拥有值
- 在这种情况下,可以去掉检查,也不必每次访问的时候都进行解包,因为它能确定每次访问的时候都有值
- 可以在类型后面加个感叹号 ! 定义一个隐式解包的可选项
let num1: Int! = 10
let num2: Int = num1
if num2 != nil {
print(num2 + 6)
}
if let num3 = num1 {
print(num3)
}
注意:隐式解包的默认值不能为nil,否则会报错
-------------------------------------------------------------------------------------------------------------------------------
字符串插值
- 可选项在字符串插值或者直接打印时,编译器会发出警告
- 至少有3种方法消除警告
-------------------------------------------------------------------------------------------------------------------------------
多重可选项
=>可转化成结构图:
我们可以看出num2等于num3
- 可以使用lldb指令 frame variable -R 或者 fr v -R查看区别
-------------------------------------------------------------------------------------------------------------------------------
多重可选项(nil情况)
=>可转化成结构图:
这里可以看出num2与num3不同:
- 可以使用lldb指令 frame variable -R 或者 fr v -R查看区别
下图我们可以看出:
表达式1 :num2不为nil,表达式转化为num1 ?? 2, num1位nil,最终得到2
表达式2 :num3为nil,表达式转化为 1 ?? 2,最终得到1。
提醒一下,可能有人认为num1 和num3同为nil应该相同,这是错的,因为两者类型不同,num1为Int?, num3 为Int??
var n = num3!! :这表示双重解包,n为Int类型。