前言:
本篇仅为视频学习笔记
★ 这个原始值,它存储到什么地方呢?(无需纠结)
enum Season: Int {
case spring = 1, summer, autumn, winter
}var s = Season.spring var s1 = Season.spring var s2 = Season.spring s2.rawValue MemoryLayout<Season>.size // 1,实际用到的空间大小 MemoryLayout<Season>.stride // 1,分配占用的空间大小 MemoryLayout<Season>.alignment // 1,对齐参数
我们把它改成上面代码那样,冒号Int。(enum Season: Int)。不就是代表枚举Season的原始值是Int类型吗,所以呢,我们写了一个原始值spring = 1。summer, autumn, winter依次默认为2、3、4。
之前,说过,这个原始值不是存储到s变量里面去的。像这些枚举变量。它只占用一个字节,它一个字节就能搞的清楚,它是spring,是summer,是autumn,还是winter。
那你可能就会想,那么这个原始值存储到哪里呢?其实这个东西,你思考一下,它不存也可以啊,怎么不存也可以呢?
因为,你想想,你拿原始值不过就是调用 s2.rawValue。那么这个rawValue,完全怎么可以实现呢?
完全可以这样实现,如下代码:
enum Season: Int { case spring = 1, summer, autumn, winter func rawValue() -> Int { if self == 0 return 1 if self == 1 return 2 if self == 2 return 3 if self == 3 return 4 } }
所以,完全可以不用思考,它存在什么地方,它完全可以不用存。
是不是可以这个样子,如果发现我自己内存里面存储的是0,也就是存储的是spring,我们就return 1。那么,如果我发现自己内存里,存储的是1,不就意味着我是summer,我是summer,那么我的原始值就是return 2。那么我这个rawValue获取原始值的实现内部完全可以如上面代码那样写。
所以,大家没有必要去纠结,如果这里写的是enum Season:String,那么你关联的这个原始值,它存储到什么地方呢?它不一定需要存储。所以,这个放哪里,这个跟你无关。你只要搞清楚,这个枚举变量占用多少内存就可以了。
★ Password枚举为什么实际用到的空间大小为33个字节呢?
enum Password { case number(Int,Int,Int,Int) // 32 case other // 1 } var pwd = Password.number(5, 6, 4, 7) // 占用32字节 pwd = .other var pwd1 = Password.number(2, 4, 3, 0) // 占用32字节 var pwd2 = Password.number(23, 43, 133,30) // 占用32字节 MemoryLayout<Password>.stride // 40,分配占用的空间大小 MemoryLayout<Password>.size // 33,实际用到的空间大小 MemoryLayout<Password>.alignment // 8,对齐参数
再说另外一个问题,刚刚说到Password这个枚举变量的话,那么我定义一个pwd枚举变量。最后真正分配的内存是多少? 是40,但实际上真正利用起来的是33。其实这个33是怎么利用起来的呢?
我们在 case number(Int,Int,Int,Int) 这个位置标记一个32,是没有问题的。但是我们在case other // 1标记一个1的话,其实不是很严谨。那么这个1,这个字节是怎么用的呢?
其实,有的人会怎么想呢?它会想,你这个pwd这个枚举变量,只占用32个字节,不就够了吗?为什么呢?首先,如果你的pwd是 case number(Int,Int,Int,Int) 这种类型的话4个证书,刚好是32个字节,存储到里面去。但是,如果你是other的话,大家有可能会想,像下面这种:
enum Password { case other, abc, ddd }
非常普通的这些成员,他是不是只需要一个字节就够了。比如说,other是0、abc是1,ddd是2。是不是用一个字节,就能把这些数值表达清楚了。那么,一说到这个,大家可能会想,那么你存储other只需要1个字节,但是在var pwd = Password.number(5, 6, 4, 7) 这行时,我们不是已经分配了32个字节给pwd了吗?
那么你的.other赋值给pwd。( pwd = .other ),就不能将这个other,1个字节的数据存储到var pwd = Password.number(5, 6, 4, 7) 这32个字节里面去吗?肯定会有这样的问题,有这样的问题说明你还没想透。
你思考一下,如果你将 .other 这一个值。假设你这个值 case other是 0。那么,你能把这个0存储到刚刚 Password.number(5, 6, 4, 7) 32个字节里面去吗?
那我就问你将来你怎么判断呢?你怎么知道,它是.otner还是 .number呢?根本就分不清。请问我现在给你32个字节,你怎么知道32个字节里面存储的是.otner这个家伙还是这(5, 6, 4, 7) 四个家伙。
因为你拿到32,完全可以把这32个字节分成4个部分,每个部分8个字节,然后分别认为是这(Int,Int,Int,Int) 四个整型。
我也可以拿到32个字节中,最前面的一个字节,当作是 .other。所以,你必须var pwd = Password.number(5, 6, 4, 7) // 32 + 1 其中的一个字节是来告诉你是case other,这个类型,还是case number(Int,Int,Int,Int) // 32 这个类型。你可以这样思考,这是用来辨别的。就是从字节中,我们可以看得出来,我们整体数据是case number(Int,Int,Int,Int) 这个家伙,还是case other这个家伙。
所以,只给它enum Password 32个字节是不行的。所以,必然是32 + 1,那么,你可能会想为什么不是 32 + 2呢?没必要这么浪费,你搞出一个来,就行了。
既然,你需要占用33个字节,再加上它的内存对齐参数是8个字节,所以最后分配给它的是40个字节。